@thinkhive/sdk 2.0.0 → 3.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.
@@ -0,0 +1,194 @@
1
+ /**
2
+ * ThinkHive SDK v3.0 - LangChain/LangGraph Instrumentation
3
+ *
4
+ * Auto-instrumentation for LangChain and LangGraph:
5
+ * - Chains
6
+ * - Agents
7
+ * - LangGraph workflows
8
+ * - Retrievers
9
+ * - Tool calls
10
+ */
11
+ import type { ConversationMessage, SpanData } from '../core/types';
12
+ /**
13
+ * LangChain callback handler for ThinkHive tracing
14
+ *
15
+ * @example
16
+ * ```typescript
17
+ * import { ChatOpenAI } from '@langchain/openai';
18
+ * import { ThinkHiveCallbackHandler } from '@thinkhive/sdk/instrumentation/langchain';
19
+ *
20
+ * const handler = new ThinkHiveCallbackHandler();
21
+ *
22
+ * const chat = new ChatOpenAI({
23
+ * callbacks: [handler],
24
+ * });
25
+ *
26
+ * await chat.invoke('Hello!');
27
+ *
28
+ * // Get collected spans
29
+ * const spans = handler.getSpans();
30
+ * ```
31
+ */
32
+ export declare class ThinkHiveCallbackHandler {
33
+ private spans;
34
+ private spanStack;
35
+ private tracer;
36
+ /**
37
+ * Called when a chain starts
38
+ */
39
+ handleChainStart(chain: {
40
+ name?: string;
41
+ id?: string[];
42
+ }, inputs: Record<string, unknown>): void;
43
+ /**
44
+ * Called when a chain ends
45
+ */
46
+ handleChainEnd(outputs: Record<string, unknown>): void;
47
+ /**
48
+ * Called when a chain errors
49
+ */
50
+ handleChainError(error: Error): void;
51
+ /**
52
+ * Called when an LLM starts
53
+ */
54
+ handleLLMStart(llm: {
55
+ name?: string;
56
+ id?: string[];
57
+ }, prompts: string[]): void;
58
+ /**
59
+ * Called when an LLM ends
60
+ */
61
+ handleLLMEnd(output: {
62
+ generations?: Array<Array<{
63
+ text?: string;
64
+ }>>;
65
+ llmOutput?: {
66
+ tokenUsage?: {
67
+ promptTokens?: number;
68
+ completionTokens?: number;
69
+ };
70
+ };
71
+ }): void;
72
+ /**
73
+ * Called when an LLM errors
74
+ */
75
+ handleLLMError(error: Error): void;
76
+ /**
77
+ * Called when a tool starts
78
+ */
79
+ handleToolStart(tool: {
80
+ name?: string;
81
+ id?: string[];
82
+ }, input: string): void;
83
+ /**
84
+ * Called when a tool ends
85
+ */
86
+ handleToolEnd(output: string): void;
87
+ /**
88
+ * Called when a tool errors
89
+ */
90
+ handleToolError(error: Error): void;
91
+ /**
92
+ * Called when a retriever starts
93
+ */
94
+ handleRetrieverStart(retriever: {
95
+ name?: string;
96
+ id?: string[];
97
+ }, query: string): void;
98
+ /**
99
+ * Called when a retriever ends
100
+ */
101
+ handleRetrieverEnd(documents: Array<{
102
+ pageContent?: string;
103
+ metadata?: Record<string, unknown>;
104
+ }>): void;
105
+ /**
106
+ * Called when a retriever errors
107
+ */
108
+ handleRetrieverError(error: Error): void;
109
+ /**
110
+ * Get collected spans
111
+ */
112
+ getSpans(): SpanData[];
113
+ /**
114
+ * Clear collected spans
115
+ */
116
+ clearSpans(): void;
117
+ /**
118
+ * Get spans as a tree (parent-child relationships)
119
+ */
120
+ getSpanTree(): SpanData[];
121
+ }
122
+ /**
123
+ * LangGraph node wrapper for tracing
124
+ *
125
+ * @example
126
+ * ```typescript
127
+ * import { StateGraph } from '@langchain/langgraph';
128
+ * import { wrapLangGraphNode } from '@thinkhive/sdk/instrumentation/langchain';
129
+ *
130
+ * const workflow = new StateGraph({ ... });
131
+ *
132
+ * workflow.addNode('agent', wrapLangGraphNode('agent', async (state) => {
133
+ * // Your agent logic
134
+ * return { messages: [...state.messages, response] };
135
+ * }));
136
+ * ```
137
+ */
138
+ export declare function wrapLangGraphNode<TState, TResult>(nodeName: string, fn: (state: TState) => Promise<TResult>): (state: TState) => Promise<TResult>;
139
+ /**
140
+ * Wrap a LangGraph workflow execution
141
+ *
142
+ * @example
143
+ * ```typescript
144
+ * import { wrapLangGraphExecution } from '@thinkhive/sdk/instrumentation/langchain';
145
+ *
146
+ * const result = await wrapLangGraphExecution(
147
+ * 'customer_support',
148
+ * () => compiledGraph.invoke({ messages: [...] })
149
+ * );
150
+ * ```
151
+ */
152
+ export declare function wrapLangGraphExecution<T>(workflowName: string, fn: () => Promise<T>, options?: {
153
+ input?: unknown;
154
+ config?: Record<string, unknown>;
155
+ }): Promise<T>;
156
+ /**
157
+ * Extract conversation messages from LangChain format
158
+ */
159
+ export declare function extractMessagesFromLangChain(messages: Array<{
160
+ _getType?: () => string;
161
+ content: string;
162
+ additional_kwargs?: Record<string, unknown>;
163
+ }>): ConversationMessage[];
164
+ /**
165
+ * Extract spans from LangSmith run tree (if available)
166
+ */
167
+ export declare function extractSpansFromRunTree(runTree: {
168
+ name: string;
169
+ run_type: string;
170
+ start_time?: number;
171
+ end_time?: number;
172
+ inputs?: Record<string, unknown>;
173
+ outputs?: Record<string, unknown>;
174
+ error?: string;
175
+ child_runs?: any[];
176
+ }): SpanData;
177
+ /**
178
+ * Create a callback handler and return it with a cleanup function
179
+ *
180
+ * @example
181
+ * ```typescript
182
+ * const { handler, getSpans, cleanup } = createThinkHiveCallback();
183
+ *
184
+ * await chain.invoke(input, { callbacks: [handler] });
185
+ *
186
+ * const spans = getSpans();
187
+ * cleanup();
188
+ * ```
189
+ */
190
+ export declare function createThinkHiveCallback(): {
191
+ handler: ThinkHiveCallbackHandler;
192
+ getSpans: () => SpanData[];
193
+ cleanup: () => void;
194
+ };
@@ -0,0 +1,429 @@
1
+ "use strict";
2
+ /**
3
+ * ThinkHive SDK v3.0 - LangChain/LangGraph Instrumentation
4
+ *
5
+ * Auto-instrumentation for LangChain and LangGraph:
6
+ * - Chains
7
+ * - Agents
8
+ * - LangGraph workflows
9
+ * - Retrievers
10
+ * - Tool calls
11
+ */
12
+ Object.defineProperty(exports, "__esModule", { value: true });
13
+ exports.ThinkHiveCallbackHandler = void 0;
14
+ exports.wrapLangGraphNode = wrapLangGraphNode;
15
+ exports.wrapLangGraphExecution = wrapLangGraphExecution;
16
+ exports.extractMessagesFromLangChain = extractMessagesFromLangChain;
17
+ exports.extractSpansFromRunTree = extractSpansFromRunTree;
18
+ exports.createThinkHiveCallback = createThinkHiveCallback;
19
+ const api_1 = require("@opentelemetry/api");
20
+ const config_1 = require("../core/config");
21
+ // ============================================================================
22
+ // LANGCHAIN CALLBACK HANDLER
23
+ // ============================================================================
24
+ /**
25
+ * LangChain callback handler for ThinkHive tracing
26
+ *
27
+ * @example
28
+ * ```typescript
29
+ * import { ChatOpenAI } from '@langchain/openai';
30
+ * import { ThinkHiveCallbackHandler } from '@thinkhive/sdk/instrumentation/langchain';
31
+ *
32
+ * const handler = new ThinkHiveCallbackHandler();
33
+ *
34
+ * const chat = new ChatOpenAI({
35
+ * callbacks: [handler],
36
+ * });
37
+ *
38
+ * await chat.invoke('Hello!');
39
+ *
40
+ * // Get collected spans
41
+ * const spans = handler.getSpans();
42
+ * ```
43
+ */
44
+ class ThinkHiveCallbackHandler {
45
+ constructor() {
46
+ this.spans = [];
47
+ this.spanStack = [];
48
+ this.tracer = api_1.trace.getTracer('thinkhive', '3.0.0');
49
+ }
50
+ /**
51
+ * Called when a chain starts
52
+ */
53
+ handleChainStart(chain, inputs) {
54
+ const span = {
55
+ name: chain.name || 'chain',
56
+ type: 'chain',
57
+ input: inputs,
58
+ status: 'ok',
59
+ };
60
+ this.spanStack.push({ span, startTime: Date.now() });
61
+ (0, config_1.debugLog)('Chain started:', chain.name);
62
+ }
63
+ /**
64
+ * Called when a chain ends
65
+ */
66
+ handleChainEnd(outputs) {
67
+ const entry = this.spanStack.pop();
68
+ if (entry) {
69
+ entry.span.durationMs = Date.now() - entry.startTime;
70
+ entry.span.output = outputs;
71
+ this.spans.push(entry.span);
72
+ (0, config_1.debugLog)('Chain ended:', entry.span.name);
73
+ }
74
+ }
75
+ /**
76
+ * Called when a chain errors
77
+ */
78
+ handleChainError(error) {
79
+ const entry = this.spanStack.pop();
80
+ if (entry) {
81
+ entry.span.durationMs = Date.now() - entry.startTime;
82
+ entry.span.status = 'error';
83
+ entry.span.error = error.message;
84
+ this.spans.push(entry.span);
85
+ (0, config_1.debugLog)('Chain error:', entry.span.name, error.message);
86
+ }
87
+ }
88
+ /**
89
+ * Called when an LLM starts
90
+ */
91
+ handleLLMStart(llm, prompts) {
92
+ const span = {
93
+ name: llm.name || 'llm',
94
+ type: 'llm',
95
+ input: prompts,
96
+ status: 'ok',
97
+ };
98
+ this.spanStack.push({ span, startTime: Date.now() });
99
+ (0, config_1.debugLog)('LLM started:', llm.name);
100
+ }
101
+ /**
102
+ * Called when an LLM ends
103
+ */
104
+ handleLLMEnd(output) {
105
+ const entry = this.spanStack.pop();
106
+ if (entry) {
107
+ entry.span.durationMs = Date.now() - entry.startTime;
108
+ entry.span.output = output.generations?.[0]?.[0]?.text;
109
+ if (output.llmOutput?.tokenUsage) {
110
+ entry.span.promptTokens = output.llmOutput.tokenUsage.promptTokens;
111
+ entry.span.completionTokens = output.llmOutput.tokenUsage.completionTokens;
112
+ }
113
+ this.spans.push(entry.span);
114
+ (0, config_1.debugLog)('LLM ended:', entry.span.name);
115
+ }
116
+ }
117
+ /**
118
+ * Called when an LLM errors
119
+ */
120
+ handleLLMError(error) {
121
+ const entry = this.spanStack.pop();
122
+ if (entry) {
123
+ entry.span.durationMs = Date.now() - entry.startTime;
124
+ entry.span.status = 'error';
125
+ entry.span.error = error.message;
126
+ this.spans.push(entry.span);
127
+ (0, config_1.debugLog)('LLM error:', entry.span.name, error.message);
128
+ }
129
+ }
130
+ /**
131
+ * Called when a tool starts
132
+ */
133
+ handleToolStart(tool, input) {
134
+ let parsedInput;
135
+ try {
136
+ parsedInput = JSON.parse(input);
137
+ }
138
+ catch {
139
+ parsedInput = input;
140
+ }
141
+ const span = {
142
+ name: tool.name || 'tool',
143
+ type: 'tool',
144
+ toolName: tool.name,
145
+ input: parsedInput,
146
+ status: 'ok',
147
+ };
148
+ this.spanStack.push({ span, startTime: Date.now() });
149
+ (0, config_1.debugLog)('Tool started:', tool.name);
150
+ }
151
+ /**
152
+ * Called when a tool ends
153
+ */
154
+ handleToolEnd(output) {
155
+ const entry = this.spanStack.pop();
156
+ if (entry) {
157
+ entry.span.durationMs = Date.now() - entry.startTime;
158
+ try {
159
+ entry.span.output = JSON.parse(output);
160
+ }
161
+ catch {
162
+ entry.span.output = output;
163
+ }
164
+ this.spans.push(entry.span);
165
+ (0, config_1.debugLog)('Tool ended:', entry.span.name);
166
+ }
167
+ }
168
+ /**
169
+ * Called when a tool errors
170
+ */
171
+ handleToolError(error) {
172
+ const entry = this.spanStack.pop();
173
+ if (entry) {
174
+ entry.span.durationMs = Date.now() - entry.startTime;
175
+ entry.span.status = 'error';
176
+ entry.span.error = error.message;
177
+ this.spans.push(entry.span);
178
+ (0, config_1.debugLog)('Tool error:', entry.span.name, error.message);
179
+ }
180
+ }
181
+ /**
182
+ * Called when a retriever starts
183
+ */
184
+ handleRetrieverStart(retriever, query) {
185
+ const span = {
186
+ name: retriever.name || 'retriever',
187
+ type: 'retrieval',
188
+ query,
189
+ status: 'ok',
190
+ };
191
+ this.spanStack.push({ span, startTime: Date.now() });
192
+ (0, config_1.debugLog)('Retriever started:', retriever.name);
193
+ }
194
+ /**
195
+ * Called when a retriever ends
196
+ */
197
+ handleRetrieverEnd(documents) {
198
+ const entry = this.spanStack.pop();
199
+ if (entry) {
200
+ entry.span.durationMs = Date.now() - entry.startTime;
201
+ entry.span.documentCount = documents.length;
202
+ entry.span.sources = documents
203
+ .map((d) => d.metadata?.source)
204
+ .filter(Boolean);
205
+ entry.span.output = documents.map((d) => d.pageContent?.substring(0, 500));
206
+ this.spans.push(entry.span);
207
+ (0, config_1.debugLog)('Retriever ended:', entry.span.name, `${documents.length} docs`);
208
+ }
209
+ }
210
+ /**
211
+ * Called when a retriever errors
212
+ */
213
+ handleRetrieverError(error) {
214
+ const entry = this.spanStack.pop();
215
+ if (entry) {
216
+ entry.span.durationMs = Date.now() - entry.startTime;
217
+ entry.span.status = 'error';
218
+ entry.span.error = error.message;
219
+ this.spans.push(entry.span);
220
+ (0, config_1.debugLog)('Retriever error:', entry.span.name, error.message);
221
+ }
222
+ }
223
+ /**
224
+ * Get collected spans
225
+ */
226
+ getSpans() {
227
+ return [...this.spans];
228
+ }
229
+ /**
230
+ * Clear collected spans
231
+ */
232
+ clearSpans() {
233
+ this.spans = [];
234
+ this.spanStack = [];
235
+ }
236
+ /**
237
+ * Get spans as a tree (parent-child relationships)
238
+ */
239
+ getSpanTree() {
240
+ // Build tree from flat list based on timing
241
+ // This is a simplified version - real implementation would use run IDs
242
+ return this.spans;
243
+ }
244
+ }
245
+ exports.ThinkHiveCallbackHandler = ThinkHiveCallbackHandler;
246
+ // ============================================================================
247
+ // LANGGRAPH TRACING
248
+ // ============================================================================
249
+ /**
250
+ * LangGraph node wrapper for tracing
251
+ *
252
+ * @example
253
+ * ```typescript
254
+ * import { StateGraph } from '@langchain/langgraph';
255
+ * import { wrapLangGraphNode } from '@thinkhive/sdk/instrumentation/langchain';
256
+ *
257
+ * const workflow = new StateGraph({ ... });
258
+ *
259
+ * workflow.addNode('agent', wrapLangGraphNode('agent', async (state) => {
260
+ * // Your agent logic
261
+ * return { messages: [...state.messages, response] };
262
+ * }));
263
+ * ```
264
+ */
265
+ function wrapLangGraphNode(nodeName, fn) {
266
+ const tracer = api_1.trace.getTracer('thinkhive', '3.0.0');
267
+ return async (state) => {
268
+ return tracer.startActiveSpan(`langgraph.node.${nodeName}`, {
269
+ attributes: {
270
+ 'openinference.span.kind': 'CHAIN',
271
+ 'langgraph.node_name': nodeName,
272
+ 'langgraph.state_keys': Object.keys(state).join(','),
273
+ },
274
+ }, async (span) => {
275
+ const startTime = Date.now();
276
+ try {
277
+ const result = await fn(state);
278
+ span.setAttribute('langgraph.result_keys', Object.keys(result).join(','));
279
+ span.setStatus({ code: api_1.SpanStatusCode.OK });
280
+ return result;
281
+ }
282
+ catch (error) {
283
+ const message = error instanceof Error ? error.message : String(error);
284
+ span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
285
+ span.recordException(error);
286
+ throw error;
287
+ }
288
+ finally {
289
+ span.setAttribute('duration_ms', Date.now() - startTime);
290
+ span.end();
291
+ }
292
+ });
293
+ };
294
+ }
295
+ /**
296
+ * Wrap a LangGraph workflow execution
297
+ *
298
+ * @example
299
+ * ```typescript
300
+ * import { wrapLangGraphExecution } from '@thinkhive/sdk/instrumentation/langchain';
301
+ *
302
+ * const result = await wrapLangGraphExecution(
303
+ * 'customer_support',
304
+ * () => compiledGraph.invoke({ messages: [...] })
305
+ * );
306
+ * ```
307
+ */
308
+ async function wrapLangGraphExecution(workflowName, fn, options = {}) {
309
+ const tracer = api_1.trace.getTracer('thinkhive', '3.0.0');
310
+ return tracer.startActiveSpan(`langgraph.workflow.${workflowName}`, {
311
+ attributes: {
312
+ 'openinference.span.kind': 'AGENT',
313
+ 'langgraph.workflow_name': workflowName,
314
+ 'langgraph.input': options.input
315
+ ? JSON.stringify(options.input).substring(0, 10000)
316
+ : undefined,
317
+ },
318
+ }, async (span) => {
319
+ const startTime = Date.now();
320
+ try {
321
+ const result = await fn();
322
+ span.setAttribute('langgraph.output', JSON.stringify(result).substring(0, 10000));
323
+ span.setStatus({ code: api_1.SpanStatusCode.OK });
324
+ return result;
325
+ }
326
+ catch (error) {
327
+ const message = error instanceof Error ? error.message : String(error);
328
+ span.setStatus({ code: api_1.SpanStatusCode.ERROR, message });
329
+ span.recordException(error);
330
+ throw error;
331
+ }
332
+ finally {
333
+ span.setAttribute('duration_ms', Date.now() - startTime);
334
+ span.end();
335
+ }
336
+ });
337
+ }
338
+ // ============================================================================
339
+ // EXTRACTION HELPERS
340
+ // ============================================================================
341
+ /**
342
+ * Extract conversation messages from LangChain format
343
+ */
344
+ function extractMessagesFromLangChain(messages) {
345
+ return messages.map((m) => {
346
+ const type = m._getType?.() || 'unknown';
347
+ let role = 'user';
348
+ switch (type) {
349
+ case 'human':
350
+ role = 'user';
351
+ break;
352
+ case 'ai':
353
+ role = 'assistant';
354
+ break;
355
+ case 'system':
356
+ role = 'system';
357
+ break;
358
+ case 'tool':
359
+ role = 'tool';
360
+ break;
361
+ }
362
+ return {
363
+ role,
364
+ content: m.content,
365
+ metadata: m.additional_kwargs,
366
+ };
367
+ });
368
+ }
369
+ /**
370
+ * Extract spans from LangSmith run tree (if available)
371
+ */
372
+ function extractSpansFromRunTree(runTree) {
373
+ const span = {
374
+ name: runTree.name,
375
+ type: mapRunTypeToSpanType(runTree.run_type),
376
+ input: runTree.inputs,
377
+ output: runTree.outputs,
378
+ durationMs: runTree.start_time && runTree.end_time
379
+ ? runTree.end_time - runTree.start_time
380
+ : undefined,
381
+ status: runTree.error ? 'error' : 'ok',
382
+ error: runTree.error,
383
+ };
384
+ if (runTree.child_runs && runTree.child_runs.length > 0) {
385
+ span.children = runTree.child_runs.map(extractSpansFromRunTree);
386
+ }
387
+ return span;
388
+ }
389
+ function mapRunTypeToSpanType(runType) {
390
+ switch (runType) {
391
+ case 'llm':
392
+ return 'llm';
393
+ case 'tool':
394
+ return 'tool';
395
+ case 'retriever':
396
+ return 'retrieval';
397
+ case 'embedding':
398
+ return 'embedding';
399
+ case 'chain':
400
+ case 'agent':
401
+ default:
402
+ return 'chain';
403
+ }
404
+ }
405
+ // ============================================================================
406
+ // CONVENIENCE FUNCTIONS
407
+ // ============================================================================
408
+ /**
409
+ * Create a callback handler and return it with a cleanup function
410
+ *
411
+ * @example
412
+ * ```typescript
413
+ * const { handler, getSpans, cleanup } = createThinkHiveCallback();
414
+ *
415
+ * await chain.invoke(input, { callbacks: [handler] });
416
+ *
417
+ * const spans = getSpans();
418
+ * cleanup();
419
+ * ```
420
+ */
421
+ function createThinkHiveCallback() {
422
+ const handler = new ThinkHiveCallbackHandler();
423
+ return {
424
+ handler,
425
+ getSpans: () => handler.getSpans(),
426
+ cleanup: () => handler.clearSpans(),
427
+ };
428
+ }
429
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"langchain.js","sourceRoot":"","sources":["../../src/instrumentation/langchain.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ThinkHive SDK v3.0 - LangChain/LangGraph Instrumentation\n *\n * Auto-instrumentation for LangChain and LangGraph:\n * - Chains\n * - Agents\n * - LangGraph workflows\n * - Retrievers\n * - Tool calls\n */\n\nimport { trace, SpanStatusCode } from '@opentelemetry/api';\nimport { debugLog } from '../core/config';\nimport type { ConversationMessage, SpanData } from '../core/types';\n\n// ============================================================================\n// LANGCHAIN CALLBACK HANDLER\n// ============================================================================\n\n/**\n * LangChain callback handler for ThinkHive tracing\n *\n * @example\n * ```typescript\n * import { ChatOpenAI } from '@langchain/openai';\n * import { ThinkHiveCallbackHandler } from '@thinkhive/sdk/instrumentation/langchain';\n *\n * const handler = new ThinkHiveCallbackHandler();\n *\n * const chat = new ChatOpenAI({\n *   callbacks: [handler],\n * });\n *\n * await chat.invoke('Hello!');\n *\n * // Get collected spans\n * const spans = handler.getSpans();\n * ```\n */\nexport class ThinkHiveCallbackHandler {\n  private spans: SpanData[] = [];\n  private spanStack: Array<{ span: SpanData; startTime: number }> = [];\n  private tracer = trace.getTracer('thinkhive', '3.0.0');\n\n  /**\n   * Called when a chain starts\n   */\n  handleChainStart(\n    chain: { name?: string; id?: string[] },\n    inputs: Record<string, unknown>\n  ): void {\n    const span: SpanData = {\n      name: chain.name || 'chain',\n      type: 'chain',\n      input: inputs,\n      status: 'ok',\n    };\n    this.spanStack.push({ span, startTime: Date.now() });\n    debugLog('Chain started:', chain.name);\n  }\n\n  /**\n   * Called when a chain ends\n   */\n  handleChainEnd(outputs: Record<string, unknown>): void {\n    const entry = this.spanStack.pop();\n    if (entry) {\n      entry.span.durationMs = Date.now() - entry.startTime;\n      entry.span.output = outputs;\n      this.spans.push(entry.span);\n      debugLog('Chain ended:', entry.span.name);\n    }\n  }\n\n  /**\n   * Called when a chain errors\n   */\n  handleChainError(error: Error): void {\n    const entry = this.spanStack.pop();\n    if (entry) {\n      entry.span.durationMs = Date.now() - entry.startTime;\n      entry.span.status = 'error';\n      entry.span.error = error.message;\n      this.spans.push(entry.span);\n      debugLog('Chain error:', entry.span.name, error.message);\n    }\n  }\n\n  /**\n   * Called when an LLM starts\n   */\n  handleLLMStart(\n    llm: { name?: string; id?: string[] },\n    prompts: string[]\n  ): void {\n    const span: SpanData = {\n      name: llm.name || 'llm',\n      type: 'llm',\n      input: prompts,\n      status: 'ok',\n    };\n    this.spanStack.push({ span, startTime: Date.now() });\n    debugLog('LLM started:', llm.name);\n  }\n\n  /**\n   * Called when an LLM ends\n   */\n  handleLLMEnd(output: {\n    generations?: Array<Array<{ text?: string }>>;\n    llmOutput?: { tokenUsage?: { promptTokens?: number; completionTokens?: number } };\n  }): void {\n    const entry = this.spanStack.pop();\n    if (entry) {\n      entry.span.durationMs = Date.now() - entry.startTime;\n      entry.span.output = output.generations?.[0]?.[0]?.text;\n      if (output.llmOutput?.tokenUsage) {\n        entry.span.promptTokens = output.llmOutput.tokenUsage.promptTokens;\n        entry.span.completionTokens = output.llmOutput.tokenUsage.completionTokens;\n      }\n      this.spans.push(entry.span);\n      debugLog('LLM ended:', entry.span.name);\n    }\n  }\n\n  /**\n   * Called when an LLM errors\n   */\n  handleLLMError(error: Error): void {\n    const entry = this.spanStack.pop();\n    if (entry) {\n      entry.span.durationMs = Date.now() - entry.startTime;\n      entry.span.status = 'error';\n      entry.span.error = error.message;\n      this.spans.push(entry.span);\n      debugLog('LLM error:', entry.span.name, error.message);\n    }\n  }\n\n  /**\n   * Called when a tool starts\n   */\n  handleToolStart(\n    tool: { name?: string; id?: string[] },\n    input: string\n  ): void {\n    let parsedInput: unknown;\n    try {\n      parsedInput = JSON.parse(input);\n    } catch {\n      parsedInput = input;\n    }\n\n    const span: SpanData = {\n      name: tool.name || 'tool',\n      type: 'tool',\n      toolName: tool.name,\n      input: parsedInput,\n      status: 'ok',\n    };\n    this.spanStack.push({ span, startTime: Date.now() });\n    debugLog('Tool started:', tool.name);\n  }\n\n  /**\n   * Called when a tool ends\n   */\n  handleToolEnd(output: string): void {\n    const entry = this.spanStack.pop();\n    if (entry) {\n      entry.span.durationMs = Date.now() - entry.startTime;\n      try {\n        entry.span.output = JSON.parse(output);\n      } catch {\n        entry.span.output = output;\n      }\n      this.spans.push(entry.span);\n      debugLog('Tool ended:', entry.span.name);\n    }\n  }\n\n  /**\n   * Called when a tool errors\n   */\n  handleToolError(error: Error): void {\n    const entry = this.spanStack.pop();\n    if (entry) {\n      entry.span.durationMs = Date.now() - entry.startTime;\n      entry.span.status = 'error';\n      entry.span.error = error.message;\n      this.spans.push(entry.span);\n      debugLog('Tool error:', entry.span.name, error.message);\n    }\n  }\n\n  /**\n   * Called when a retriever starts\n   */\n  handleRetrieverStart(\n    retriever: { name?: string; id?: string[] },\n    query: string\n  ): void {\n    const span: SpanData = {\n      name: retriever.name || 'retriever',\n      type: 'retrieval',\n      query,\n      status: 'ok',\n    };\n    this.spanStack.push({ span, startTime: Date.now() });\n    debugLog('Retriever started:', retriever.name);\n  }\n\n  /**\n   * Called when a retriever ends\n   */\n  handleRetrieverEnd(\n    documents: Array<{ pageContent?: string; metadata?: Record<string, unknown> }>\n  ): void {\n    const entry = this.spanStack.pop();\n    if (entry) {\n      entry.span.durationMs = Date.now() - entry.startTime;\n      entry.span.documentCount = documents.length;\n      entry.span.sources = documents\n        .map((d) => d.metadata?.source as string)\n        .filter(Boolean);\n      entry.span.output = documents.map((d) => d.pageContent?.substring(0, 500));\n      this.spans.push(entry.span);\n      debugLog('Retriever ended:', entry.span.name, `${documents.length} docs`);\n    }\n  }\n\n  /**\n   * Called when a retriever errors\n   */\n  handleRetrieverError(error: Error): void {\n    const entry = this.spanStack.pop();\n    if (entry) {\n      entry.span.durationMs = Date.now() - entry.startTime;\n      entry.span.status = 'error';\n      entry.span.error = error.message;\n      this.spans.push(entry.span);\n      debugLog('Retriever error:', entry.span.name, error.message);\n    }\n  }\n\n  /**\n   * Get collected spans\n   */\n  getSpans(): SpanData[] {\n    return [...this.spans];\n  }\n\n  /**\n   * Clear collected spans\n   */\n  clearSpans(): void {\n    this.spans = [];\n    this.spanStack = [];\n  }\n\n  /**\n   * Get spans as a tree (parent-child relationships)\n   */\n  getSpanTree(): SpanData[] {\n    // Build tree from flat list based on timing\n    // This is a simplified version - real implementation would use run IDs\n    return this.spans;\n  }\n}\n\n// ============================================================================\n// LANGGRAPH TRACING\n// ============================================================================\n\n/**\n * LangGraph node wrapper for tracing\n *\n * @example\n * ```typescript\n * import { StateGraph } from '@langchain/langgraph';\n * import { wrapLangGraphNode } from '@thinkhive/sdk/instrumentation/langchain';\n *\n * const workflow = new StateGraph({ ... });\n *\n * workflow.addNode('agent', wrapLangGraphNode('agent', async (state) => {\n *   // Your agent logic\n *   return { messages: [...state.messages, response] };\n * }));\n * ```\n */\nexport function wrapLangGraphNode<TState, TResult>(\n  nodeName: string,\n  fn: (state: TState) => Promise<TResult>\n): (state: TState) => Promise<TResult> {\n  const tracer = trace.getTracer('thinkhive', '3.0.0');\n\n  return async (state: TState): Promise<TResult> => {\n    return tracer.startActiveSpan(\n      `langgraph.node.${nodeName}`,\n      {\n        attributes: {\n          'openinference.span.kind': 'CHAIN',\n          'langgraph.node_name': nodeName,\n          'langgraph.state_keys': Object.keys(state as object).join(','),\n        },\n      },\n      async (span) => {\n        const startTime = Date.now();\n        try {\n          const result = await fn(state);\n          span.setAttribute(\n            'langgraph.result_keys',\n            Object.keys(result as object).join(',')\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/**\n * Wrap a LangGraph workflow execution\n *\n * @example\n * ```typescript\n * import { wrapLangGraphExecution } from '@thinkhive/sdk/instrumentation/langchain';\n *\n * const result = await wrapLangGraphExecution(\n *   'customer_support',\n *   () => compiledGraph.invoke({ messages: [...] })\n * );\n * ```\n */\nexport async function wrapLangGraphExecution<T>(\n  workflowName: string,\n  fn: () => Promise<T>,\n  options: {\n    input?: unknown;\n    config?: Record<string, unknown>;\n  } = {}\n): Promise<T> {\n  const tracer = trace.getTracer('thinkhive', '3.0.0');\n\n  return tracer.startActiveSpan(\n    `langgraph.workflow.${workflowName}`,\n    {\n      attributes: {\n        'openinference.span.kind': 'AGENT',\n        'langgraph.workflow_name': workflowName,\n        'langgraph.input': options.input\n          ? JSON.stringify(options.input).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(\n          'langgraph.output',\n          JSON.stringify(result).substring(0, 10000)\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// EXTRACTION HELPERS\n// ============================================================================\n\n/**\n * Extract conversation messages from LangChain format\n */\nexport function extractMessagesFromLangChain(\n  messages: Array<{\n    _getType?: () => string;\n    content: string;\n    additional_kwargs?: Record<string, unknown>;\n  }>\n): ConversationMessage[] {\n  return messages.map((m) => {\n    const type = m._getType?.() || 'unknown';\n    let role: ConversationMessage['role'] = 'user';\n\n    switch (type) {\n      case 'human':\n        role = 'user';\n        break;\n      case 'ai':\n        role = 'assistant';\n        break;\n      case 'system':\n        role = 'system';\n        break;\n      case 'tool':\n        role = 'tool';\n        break;\n    }\n\n    return {\n      role,\n      content: m.content,\n      metadata: m.additional_kwargs,\n    };\n  });\n}\n\n/**\n * Extract spans from LangSmith run tree (if available)\n */\nexport function extractSpansFromRunTree(\n  runTree: {\n    name: string;\n    run_type: string;\n    start_time?: number;\n    end_time?: number;\n    inputs?: Record<string, unknown>;\n    outputs?: Record<string, unknown>;\n    error?: string;\n    child_runs?: any[];\n  }\n): SpanData {\n  const span: SpanData = {\n    name: runTree.name,\n    type: mapRunTypeToSpanType(runTree.run_type),\n    input: runTree.inputs,\n    output: runTree.outputs,\n    durationMs:\n      runTree.start_time && runTree.end_time\n        ? runTree.end_time - runTree.start_time\n        : undefined,\n    status: runTree.error ? 'error' : 'ok',\n    error: runTree.error,\n  };\n\n  if (runTree.child_runs && runTree.child_runs.length > 0) {\n    span.children = runTree.child_runs.map(extractSpansFromRunTree);\n  }\n\n  return span;\n}\n\nfunction mapRunTypeToSpanType(runType: string): SpanData['type'] {\n  switch (runType) {\n    case 'llm':\n      return 'llm';\n    case 'tool':\n      return 'tool';\n    case 'retriever':\n      return 'retrieval';\n    case 'embedding':\n      return 'embedding';\n    case 'chain':\n    case 'agent':\n    default:\n      return 'chain';\n  }\n}\n\n// ============================================================================\n// CONVENIENCE FUNCTIONS\n// ============================================================================\n\n/**\n * Create a callback handler and return it with a cleanup function\n *\n * @example\n * ```typescript\n * const { handler, getSpans, cleanup } = createThinkHiveCallback();\n *\n * await chain.invoke(input, { callbacks: [handler] });\n *\n * const spans = getSpans();\n * cleanup();\n * ```\n */\nexport function createThinkHiveCallback(): {\n  handler: ThinkHiveCallbackHandler;\n  getSpans: () => SpanData[];\n  cleanup: () => void;\n} {\n  const handler = new ThinkHiveCallbackHandler();\n\n  return {\n    handler,\n    getSpans: () => handler.getSpans(),\n    cleanup: () => handler.clearSpans(),\n  };\n}\n"]}