@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.
- package/MIGRATION.md +274 -0
- 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,259 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* ThinkHive SDK v3.0 - Ticket Linking
|
|
4
|
+
*
|
|
5
|
+
* Deterministic linking between runs and support tickets
|
|
6
|
+
* 7 link methods with explicit confidence scores
|
|
7
|
+
*/
|
|
8
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
9
|
+
exports.linking = exports.LINK_METHOD_CONFIDENCE = void 0;
|
|
10
|
+
exports.generateZendeskMarker = generateZendeskMarker;
|
|
11
|
+
exports.parseZendeskMarker = parseZendeskMarker;
|
|
12
|
+
exports.hasZendeskMarker = hasZendeskMarker;
|
|
13
|
+
exports.removeZendeskMarker = removeZendeskMarker;
|
|
14
|
+
exports.linkRunToTicket = linkRunToTicket;
|
|
15
|
+
exports.linkRunToZendeskTicket = linkRunToZendeskTicket;
|
|
16
|
+
exports.getBestLinkMethod = getBestLinkMethod;
|
|
17
|
+
exports.formatLinkConfidence = formatLinkConfidence;
|
|
18
|
+
exports.getConfidenceLevel = getConfidenceLevel;
|
|
19
|
+
const client_1 = require("../core/client");
|
|
20
|
+
// ============================================================================
|
|
21
|
+
// ZENDESK MARKER
|
|
22
|
+
// ============================================================================
|
|
23
|
+
/**
|
|
24
|
+
* Zendesk marker format: [THID:run_xxx]
|
|
25
|
+
* Embedded in agent responses for deterministic linking
|
|
26
|
+
*/
|
|
27
|
+
const MARKER_PREFIX = '[THID:';
|
|
28
|
+
const MARKER_SUFFIX = ']';
|
|
29
|
+
const MARKER_REGEX = /\[THID:([a-zA-Z0-9_-]+)\]/;
|
|
30
|
+
/**
|
|
31
|
+
* Generate a Zendesk marker to embed in agent responses
|
|
32
|
+
*
|
|
33
|
+
* @example
|
|
34
|
+
* ```typescript
|
|
35
|
+
* const runId = 'run_abc123';
|
|
36
|
+
* const marker = generateZendeskMarker(runId);
|
|
37
|
+
* // Returns: '[THID:run_abc123]'
|
|
38
|
+
*
|
|
39
|
+
* // Append to your agent response:
|
|
40
|
+
* const response = `I've found your order. ${marker}`;
|
|
41
|
+
* ```
|
|
42
|
+
*/
|
|
43
|
+
function generateZendeskMarker(runId) {
|
|
44
|
+
if (!runId) {
|
|
45
|
+
throw new Error('Run ID is required to generate Zendesk marker');
|
|
46
|
+
}
|
|
47
|
+
return `${MARKER_PREFIX}${runId}${MARKER_SUFFIX}`;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Parse a Zendesk marker from text
|
|
51
|
+
*
|
|
52
|
+
* @example
|
|
53
|
+
* ```typescript
|
|
54
|
+
* const text = 'Thank you for contacting us. [THID:run_abc123]';
|
|
55
|
+
* const runId = parseZendeskMarker(text);
|
|
56
|
+
* // Returns: 'run_abc123'
|
|
57
|
+
* ```
|
|
58
|
+
*/
|
|
59
|
+
function parseZendeskMarker(text) {
|
|
60
|
+
const match = text.match(MARKER_REGEX);
|
|
61
|
+
return match ? match[1] : null;
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Check if text contains a Zendesk marker
|
|
65
|
+
*/
|
|
66
|
+
function hasZendeskMarker(text) {
|
|
67
|
+
return MARKER_REGEX.test(text);
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Remove Zendesk marker from text (for clean display)
|
|
71
|
+
*
|
|
72
|
+
* @example
|
|
73
|
+
* ```typescript
|
|
74
|
+
* const text = 'Thank you! [THID:run_abc123]';
|
|
75
|
+
* const clean = removeZendeskMarker(text);
|
|
76
|
+
* // Returns: 'Thank you!'
|
|
77
|
+
* ```
|
|
78
|
+
*/
|
|
79
|
+
function removeZendeskMarker(text) {
|
|
80
|
+
return text.replace(MARKER_REGEX, '').trim();
|
|
81
|
+
}
|
|
82
|
+
// ============================================================================
|
|
83
|
+
// LINK METHODS & CONFIDENCE
|
|
84
|
+
// ============================================================================
|
|
85
|
+
/**
|
|
86
|
+
* Confidence scores for each link method
|
|
87
|
+
* Based on deterministic linking principles from v3 spec
|
|
88
|
+
*/
|
|
89
|
+
exports.LINK_METHOD_CONFIDENCE = {
|
|
90
|
+
sdk_explicit: 1.0, // Direct SDK call with ticket ID
|
|
91
|
+
zendesk_marker: 1.0, // Embedded THID marker in response
|
|
92
|
+
custom_field: 1.0, // Zendesk custom field
|
|
93
|
+
middleware_stamp: 0.98, // Middleware-injected trace ID
|
|
94
|
+
session_match: 0.95, // Session ID correlation
|
|
95
|
+
email_time_window: 0.6, // Email + 15min window (lowest confidence)
|
|
96
|
+
manual: 1.0, // Human-assigned
|
|
97
|
+
};
|
|
98
|
+
/**
|
|
99
|
+
* Linking API client
|
|
100
|
+
*/
|
|
101
|
+
exports.linking = {
|
|
102
|
+
/**
|
|
103
|
+
* Create a link between a run and a ticket
|
|
104
|
+
*
|
|
105
|
+
* @example
|
|
106
|
+
* ```typescript
|
|
107
|
+
* // SDK explicit linking (highest confidence)
|
|
108
|
+
* const link = await linking.create({
|
|
109
|
+
* runId: 'run_abc123',
|
|
110
|
+
* ticketId: 'ticket_xyz',
|
|
111
|
+
* method: 'sdk_explicit',
|
|
112
|
+
* });
|
|
113
|
+
*
|
|
114
|
+
* // Zendesk marker linking
|
|
115
|
+
* const link = await linking.create({
|
|
116
|
+
* runId: 'run_abc123',
|
|
117
|
+
* externalTicketId: '12345',
|
|
118
|
+
* platform: 'zendesk',
|
|
119
|
+
* method: 'zendesk_marker',
|
|
120
|
+
* });
|
|
121
|
+
* ```
|
|
122
|
+
*/
|
|
123
|
+
async create(input) {
|
|
124
|
+
return (0, client_1.apiRequestWithData)('/links', {
|
|
125
|
+
method: 'POST',
|
|
126
|
+
body: {
|
|
127
|
+
...input,
|
|
128
|
+
confidence: exports.LINK_METHOD_CONFIDENCE[input.method],
|
|
129
|
+
},
|
|
130
|
+
});
|
|
131
|
+
},
|
|
132
|
+
/**
|
|
133
|
+
* Get links for a run
|
|
134
|
+
*
|
|
135
|
+
* @example
|
|
136
|
+
* ```typescript
|
|
137
|
+
* const links = await linking.getForRun('run_abc123');
|
|
138
|
+
* ```
|
|
139
|
+
*/
|
|
140
|
+
async getForRun(runId) {
|
|
141
|
+
return (0, client_1.apiRequestWithData)(`/runs/${runId}/links`);
|
|
142
|
+
},
|
|
143
|
+
/**
|
|
144
|
+
* Get links for a ticket
|
|
145
|
+
*
|
|
146
|
+
* @example
|
|
147
|
+
* ```typescript
|
|
148
|
+
* const links = await linking.getForTicket('ticket_xyz');
|
|
149
|
+
* ```
|
|
150
|
+
*/
|
|
151
|
+
async getForTicket(ticketId) {
|
|
152
|
+
return (0, client_1.apiRequestWithData)(`/tickets/${ticketId}/links`);
|
|
153
|
+
},
|
|
154
|
+
/**
|
|
155
|
+
* Verify a link (confirm or reject)
|
|
156
|
+
*
|
|
157
|
+
* @example
|
|
158
|
+
* ```typescript
|
|
159
|
+
* await linking.verify('link_abc123', {
|
|
160
|
+
* verified: true,
|
|
161
|
+
* notes: 'Confirmed by support agent',
|
|
162
|
+
* });
|
|
163
|
+
* ```
|
|
164
|
+
*/
|
|
165
|
+
async verify(linkId, options) {
|
|
166
|
+
return (0, client_1.apiRequestWithData)(`/links/${linkId}/verify`, {
|
|
167
|
+
method: 'POST',
|
|
168
|
+
body: options,
|
|
169
|
+
});
|
|
170
|
+
},
|
|
171
|
+
/**
|
|
172
|
+
* Delete a link
|
|
173
|
+
*
|
|
174
|
+
* @example
|
|
175
|
+
* ```typescript
|
|
176
|
+
* await linking.delete('link_abc123');
|
|
177
|
+
* ```
|
|
178
|
+
*/
|
|
179
|
+
async delete(linkId) {
|
|
180
|
+
await (0, client_1.apiRequest)(`/links/${linkId}`, { method: 'DELETE' });
|
|
181
|
+
},
|
|
182
|
+
/**
|
|
183
|
+
* Auto-link runs to tickets based on available evidence
|
|
184
|
+
*
|
|
185
|
+
* @example
|
|
186
|
+
* ```typescript
|
|
187
|
+
* const results = await linking.autoLink('run_abc123');
|
|
188
|
+
* for (const link of results.created) {
|
|
189
|
+
* console.log(`Linked to ${link.ticketId} via ${link.method}`);
|
|
190
|
+
* }
|
|
191
|
+
* ```
|
|
192
|
+
*/
|
|
193
|
+
async autoLink(runId) {
|
|
194
|
+
return (0, client_1.apiRequestWithData)(`/runs/${runId}/auto-link`, {
|
|
195
|
+
method: 'POST',
|
|
196
|
+
});
|
|
197
|
+
},
|
|
198
|
+
};
|
|
199
|
+
// ============================================================================
|
|
200
|
+
// HELPER FUNCTIONS
|
|
201
|
+
// ============================================================================
|
|
202
|
+
/**
|
|
203
|
+
* Create SDK explicit link (convenience function)
|
|
204
|
+
*/
|
|
205
|
+
async function linkRunToTicket(runId, ticketId) {
|
|
206
|
+
return exports.linking.create({
|
|
207
|
+
runId,
|
|
208
|
+
ticketId,
|
|
209
|
+
method: 'sdk_explicit',
|
|
210
|
+
});
|
|
211
|
+
}
|
|
212
|
+
/**
|
|
213
|
+
* Create Zendesk link via marker
|
|
214
|
+
*/
|
|
215
|
+
async function linkRunToZendeskTicket(runId, zendeskTicketId) {
|
|
216
|
+
return exports.linking.create({
|
|
217
|
+
runId,
|
|
218
|
+
externalTicketId: zendeskTicketId,
|
|
219
|
+
platform: 'zendesk',
|
|
220
|
+
method: 'zendesk_marker',
|
|
221
|
+
});
|
|
222
|
+
}
|
|
223
|
+
/**
|
|
224
|
+
* Get the best link method for a given scenario
|
|
225
|
+
*/
|
|
226
|
+
function getBestLinkMethod(available) {
|
|
227
|
+
if (available.hasTicketId)
|
|
228
|
+
return 'sdk_explicit';
|
|
229
|
+
if (available.hasMarker)
|
|
230
|
+
return 'zendesk_marker';
|
|
231
|
+
if (available.hasCustomField)
|
|
232
|
+
return 'custom_field';
|
|
233
|
+
if (available.hasMiddlewareStamp)
|
|
234
|
+
return 'middleware_stamp';
|
|
235
|
+
if (available.hasSessionId)
|
|
236
|
+
return 'session_match';
|
|
237
|
+
if (available.hasEmail)
|
|
238
|
+
return 'email_time_window';
|
|
239
|
+
return null;
|
|
240
|
+
}
|
|
241
|
+
/**
|
|
242
|
+
* Format link confidence for display
|
|
243
|
+
*/
|
|
244
|
+
function formatLinkConfidence(confidence) {
|
|
245
|
+
return `${Math.round(confidence * 100)}%`;
|
|
246
|
+
}
|
|
247
|
+
/**
|
|
248
|
+
* Get confidence level label
|
|
249
|
+
*/
|
|
250
|
+
function getConfidenceLevel(confidence) {
|
|
251
|
+
if (confidence >= 1.0)
|
|
252
|
+
return 'definitive';
|
|
253
|
+
if (confidence >= 0.95)
|
|
254
|
+
return 'high';
|
|
255
|
+
if (confidence >= 0.8)
|
|
256
|
+
return 'medium';
|
|
257
|
+
return 'low';
|
|
258
|
+
}
|
|
259
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ticket-linking.js","sourceRoot":"","sources":["../../src/integrations/ticket-linking.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ThinkHive SDK v3.0 - Ticket Linking\n *\n * Deterministic linking between runs and support tickets\n * 7 link methods with explicit confidence scores\n */\n\nimport { apiRequest, apiRequestWithData } from '../core/client';\nimport type { LinkMethod } from '../core/types';\n\n// ============================================================================\n// ZENDESK MARKER\n// ============================================================================\n\n/**\n * Zendesk marker format: [THID:run_xxx]\n * Embedded in agent responses for deterministic linking\n */\nconst MARKER_PREFIX = '[THID:';\nconst MARKER_SUFFIX = ']';\nconst MARKER_REGEX = /\\[THID:([a-zA-Z0-9_-]+)\\]/;\n\n/**\n * Generate a Zendesk marker to embed in agent responses\n *\n * @example\n * ```typescript\n * const runId = 'run_abc123';\n * const marker = generateZendeskMarker(runId);\n * // Returns: '[THID:run_abc123]'\n *\n * // Append to your agent response:\n * const response = `I've found your order. ${marker}`;\n * ```\n */\nexport function generateZendeskMarker(runId: string): string {\n  if (!runId) {\n    throw new Error('Run ID is required to generate Zendesk marker');\n  }\n  return `${MARKER_PREFIX}${runId}${MARKER_SUFFIX}`;\n}\n\n/**\n * Parse a Zendesk marker from text\n *\n * @example\n * ```typescript\n * const text = 'Thank you for contacting us. [THID:run_abc123]';\n * const runId = parseZendeskMarker(text);\n * // Returns: 'run_abc123'\n * ```\n */\nexport function parseZendeskMarker(text: string): string | null {\n  const match = text.match(MARKER_REGEX);\n  return match ? match[1] : null;\n}\n\n/**\n * Check if text contains a Zendesk marker\n */\nexport function hasZendeskMarker(text: string): boolean {\n  return MARKER_REGEX.test(text);\n}\n\n/**\n * Remove Zendesk marker from text (for clean display)\n *\n * @example\n * ```typescript\n * const text = 'Thank you! [THID:run_abc123]';\n * const clean = removeZendeskMarker(text);\n * // Returns: 'Thank you!'\n * ```\n */\nexport function removeZendeskMarker(text: string): string {\n  return text.replace(MARKER_REGEX, '').trim();\n}\n\n// ============================================================================\n// LINK METHODS & CONFIDENCE\n// ============================================================================\n\n/**\n * Confidence scores for each link method\n * Based on deterministic linking principles from v3 spec\n */\nexport const LINK_METHOD_CONFIDENCE: Record<LinkMethod, number> = {\n  sdk_explicit: 1.0,      // Direct SDK call with ticket ID\n  zendesk_marker: 1.0,    // Embedded THID marker in response\n  custom_field: 1.0,      // Zendesk custom field\n  middleware_stamp: 0.98, // Middleware-injected trace ID\n  session_match: 0.95,    // Session ID correlation\n  email_time_window: 0.6, // Email + 15min window (lowest confidence)\n  manual: 1.0,            // Human-assigned\n};\n\n/**\n * Link evidence structure\n */\nexport interface LinkEvidence {\n  method: LinkMethod;\n  confidence: number;\n  timestamp: string;\n  details: {\n    runId?: string;\n    ticketId?: string;\n    externalTicketId?: string;\n    platform?: string;\n    sessionId?: string;\n    email?: string;\n    timeWindowMinutes?: number;\n    customFieldName?: string;\n    customFieldValue?: string;\n  };\n}\n\n// ============================================================================\n// LINKING API\n// ============================================================================\n\n/**\n * Create link input\n */\nexport interface CreateLinkInput {\n  runId: string;\n  ticketId?: string;\n  externalTicketId?: string;\n  platform?: 'zendesk' | 'intercom' | 'salesforce' | 'freshdesk';\n  method: LinkMethod;\n  evidence?: Partial<LinkEvidence['details']>;\n}\n\n/**\n * Link response\n */\nexport interface LinkResponse {\n  id: string;\n  runId: string;\n  ticketId?: string;\n  externalTicketId?: string;\n  platform?: string;\n  method: LinkMethod;\n  confidence: number;\n  evidence: LinkEvidence;\n  createdAt: string;\n}\n\n/**\n * Linking API client\n */\nexport const linking = {\n  /**\n   * Create a link between a run and a ticket\n   *\n   * @example\n   * ```typescript\n   * // SDK explicit linking (highest confidence)\n   * const link = await linking.create({\n   *   runId: 'run_abc123',\n   *   ticketId: 'ticket_xyz',\n   *   method: 'sdk_explicit',\n   * });\n   *\n   * // Zendesk marker linking\n   * const link = await linking.create({\n   *   runId: 'run_abc123',\n   *   externalTicketId: '12345',\n   *   platform: 'zendesk',\n   *   method: 'zendesk_marker',\n   * });\n   * ```\n   */\n  async create(input: CreateLinkInput): Promise<LinkResponse> {\n    return apiRequestWithData<LinkResponse>('/links', {\n      method: 'POST',\n      body: {\n        ...input,\n        confidence: LINK_METHOD_CONFIDENCE[input.method],\n      },\n    });\n  },\n\n  /**\n   * Get links for a run\n   *\n   * @example\n   * ```typescript\n   * const links = await linking.getForRun('run_abc123');\n   * ```\n   */\n  async getForRun(runId: string): Promise<LinkResponse[]> {\n    return apiRequestWithData<LinkResponse[]>(`/runs/${runId}/links`);\n  },\n\n  /**\n   * Get links for a ticket\n   *\n   * @example\n   * ```typescript\n   * const links = await linking.getForTicket('ticket_xyz');\n   * ```\n   */\n  async getForTicket(ticketId: string): Promise<LinkResponse[]> {\n    return apiRequestWithData<LinkResponse[]>(`/tickets/${ticketId}/links`);\n  },\n\n  /**\n   * Verify a link (confirm or reject)\n   *\n   * @example\n   * ```typescript\n   * await linking.verify('link_abc123', {\n   *   verified: true,\n   *   notes: 'Confirmed by support agent',\n   * });\n   * ```\n   */\n  async verify(\n    linkId: string,\n    options: { verified: boolean; notes?: string }\n  ): Promise<{ linkId: string; verified: boolean; message: string }> {\n    return apiRequestWithData(`/links/${linkId}/verify`, {\n      method: 'POST',\n      body: options,\n    });\n  },\n\n  /**\n   * Delete a link\n   *\n   * @example\n   * ```typescript\n   * await linking.delete('link_abc123');\n   * ```\n   */\n  async delete(linkId: string): Promise<void> {\n    await apiRequest(`/links/${linkId}`, { method: 'DELETE' });\n  },\n\n  /**\n   * Auto-link runs to tickets based on available evidence\n   *\n   * @example\n   * ```typescript\n   * const results = await linking.autoLink('run_abc123');\n   * for (const link of results.created) {\n   *   console.log(`Linked to ${link.ticketId} via ${link.method}`);\n   * }\n   * ```\n   */\n  async autoLink(runId: string): Promise<{\n    runId: string;\n    created: LinkResponse[];\n    candidates: Array<{\n      ticketId: string;\n      method: LinkMethod;\n      confidence: number;\n      reason: string;\n    }>;\n  }> {\n    return apiRequestWithData(`/runs/${runId}/auto-link`, {\n      method: 'POST',\n    });\n  },\n};\n\n// ============================================================================\n// HELPER FUNCTIONS\n// ============================================================================\n\n/**\n * Create SDK explicit link (convenience function)\n */\nexport async function linkRunToTicket(\n  runId: string,\n  ticketId: string\n): Promise<LinkResponse> {\n  return linking.create({\n    runId,\n    ticketId,\n    method: 'sdk_explicit',\n  });\n}\n\n/**\n * Create Zendesk link via marker\n */\nexport async function linkRunToZendeskTicket(\n  runId: string,\n  zendeskTicketId: string\n): Promise<LinkResponse> {\n  return linking.create({\n    runId,\n    externalTicketId: zendeskTicketId,\n    platform: 'zendesk',\n    method: 'zendesk_marker',\n  });\n}\n\n/**\n * Get the best link method for a given scenario\n */\nexport function getBestLinkMethod(\n  available: {\n    hasTicketId: boolean;\n    hasMarker: boolean;\n    hasCustomField: boolean;\n    hasMiddlewareStamp: boolean;\n    hasSessionId: boolean;\n    hasEmail: boolean;\n  }\n): LinkMethod | null {\n  if (available.hasTicketId) return 'sdk_explicit';\n  if (available.hasMarker) return 'zendesk_marker';\n  if (available.hasCustomField) return 'custom_field';\n  if (available.hasMiddlewareStamp) return 'middleware_stamp';\n  if (available.hasSessionId) return 'session_match';\n  if (available.hasEmail) return 'email_time_window';\n  return null;\n}\n\n/**\n * Format link confidence for display\n */\nexport function formatLinkConfidence(confidence: number): string {\n  return `${Math.round(confidence * 100)}%`;\n}\n\n/**\n * Get confidence level label\n */\nexport function getConfidenceLevel(\n  confidence: number\n): 'definitive' | 'high' | 'medium' | 'low' {\n  if (confidence >= 1.0) return 'definitive';\n  if (confidence >= 0.95) return 'high';\n  if (confidence >= 0.8) return 'medium';\n  return 'low';\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,19 +1,45 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@thinkhive/sdk",
|
|
3
|
-
"version": "
|
|
4
|
-
"
|
|
5
|
-
"access": "public"
|
|
6
|
-
},
|
|
7
|
-
"description": "ThinkHive SDK - AI Agent Observability Platform with explainability, RAG evaluation, and business impact analysis",
|
|
3
|
+
"version": "3.0.0",
|
|
4
|
+
"description": "ThinkHive SDK v3.0 - Run-centric AI agent observability with facts vs inferences, deterministic linking, and calibrated predictions",
|
|
8
5
|
"main": "dist/index.js",
|
|
9
6
|
"types": "dist/index.d.ts",
|
|
7
|
+
"exports": {
|
|
8
|
+
".": {
|
|
9
|
+
"require": "./dist/index.js",
|
|
10
|
+
"types": "./dist/index.d.ts"
|
|
11
|
+
},
|
|
12
|
+
"./instrumentation/openai": {
|
|
13
|
+
"require": "./dist/instrumentation/openai.js",
|
|
14
|
+
"types": "./dist/instrumentation/openai.d.ts"
|
|
15
|
+
},
|
|
16
|
+
"./instrumentation/langchain": {
|
|
17
|
+
"require": "./dist/instrumentation/langchain.js",
|
|
18
|
+
"types": "./dist/instrumentation/langchain.d.ts"
|
|
19
|
+
},
|
|
20
|
+
"./integrations/ticket-linking": {
|
|
21
|
+
"require": "./dist/integrations/ticket-linking.js",
|
|
22
|
+
"types": "./dist/integrations/ticket-linking.d.ts"
|
|
23
|
+
},
|
|
24
|
+
"./integrations/customer-context": {
|
|
25
|
+
"require": "./dist/integrations/customer-context.js",
|
|
26
|
+
"types": "./dist/integrations/customer-context.d.ts"
|
|
27
|
+
}
|
|
28
|
+
},
|
|
10
29
|
"files": [
|
|
11
30
|
"dist",
|
|
12
31
|
"README.md",
|
|
13
|
-
"LICENSE"
|
|
32
|
+
"LICENSE",
|
|
33
|
+
"MIGRATION.md"
|
|
14
34
|
],
|
|
15
35
|
"scripts": {
|
|
16
|
-
"
|
|
36
|
+
"build": "tsc",
|
|
37
|
+
"clean": "rm -rf dist",
|
|
38
|
+
"prepublishOnly": "npm run clean && npm run build",
|
|
39
|
+
"test": "vitest run",
|
|
40
|
+
"test:watch": "vitest",
|
|
41
|
+
"lint": "eslint src --ext .ts",
|
|
42
|
+
"typecheck": "tsc --noEmit"
|
|
17
43
|
},
|
|
18
44
|
"keywords": [
|
|
19
45
|
"thinkhive",
|
|
@@ -27,9 +53,15 @@
|
|
|
27
53
|
"hallucination-detection",
|
|
28
54
|
"business-impact",
|
|
29
55
|
"langchain",
|
|
56
|
+
"langgraph",
|
|
30
57
|
"openai",
|
|
31
58
|
"anthropic",
|
|
32
|
-
"agent"
|
|
59
|
+
"agent",
|
|
60
|
+
"facts-vs-inferences",
|
|
61
|
+
"calibration",
|
|
62
|
+
"zendesk",
|
|
63
|
+
"intercom",
|
|
64
|
+
"customer-support"
|
|
33
65
|
],
|
|
34
66
|
"author": "ThinkHive <support@thinkhive.ai>",
|
|
35
67
|
"license": "MIT",
|
|
@@ -42,7 +74,7 @@
|
|
|
42
74
|
"url": "https://github.com/thinkhive/thinkhive-js/issues"
|
|
43
75
|
},
|
|
44
76
|
"engines": {
|
|
45
|
-
"node": ">=
|
|
77
|
+
"node": ">=18.0.0"
|
|
46
78
|
},
|
|
47
79
|
"dependencies": {
|
|
48
80
|
"@opentelemetry/api": "^1.8.0",
|
|
@@ -50,5 +82,25 @@
|
|
|
50
82
|
"@opentelemetry/resources": "^1.21.0",
|
|
51
83
|
"@opentelemetry/sdk-trace-node": "^1.21.0",
|
|
52
84
|
"@opentelemetry/sdk-trace-base": "^1.21.0"
|
|
85
|
+
},
|
|
86
|
+
"devDependencies": {
|
|
87
|
+
"@types/node": "^20.0.0",
|
|
88
|
+
"typescript": "^5.0.0",
|
|
89
|
+
"vitest": "^1.0.0",
|
|
90
|
+
"eslint": "^8.0.0",
|
|
91
|
+
"@typescript-eslint/eslint-plugin": "^6.0.0",
|
|
92
|
+
"@typescript-eslint/parser": "^6.0.0"
|
|
93
|
+
},
|
|
94
|
+
"peerDependencies": {
|
|
95
|
+
"openai": ">=4.0.0",
|
|
96
|
+
"@langchain/core": ">=0.1.0"
|
|
97
|
+
},
|
|
98
|
+
"peerDependenciesMeta": {
|
|
99
|
+
"openai": {
|
|
100
|
+
"optional": true
|
|
101
|
+
},
|
|
102
|
+
"@langchain/core": {
|
|
103
|
+
"optional": true
|
|
104
|
+
}
|
|
53
105
|
}
|
|
54
106
|
}
|