@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,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"]}
|