@thinkhive/sdk 3.1.0 → 3.3.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/README.md +279 -128
- package/dist/api/apiKeys.d.ts +252 -0
- package/dist/api/apiKeys.js +298 -0
- package/dist/api/business-metrics.d.ts +188 -0
- package/dist/api/business-metrics.js +213 -0
- package/dist/api/conversation-eval.d.ts +200 -0
- package/dist/api/conversation-eval.js +235 -0
- package/dist/api/deterministic-graders.d.ts +205 -0
- package/dist/api/deterministic-graders.js +191 -0
- package/dist/api/eval-health.d.ts +250 -0
- package/dist/api/eval-health.js +224 -0
- package/dist/api/human-review.d.ts +275 -0
- package/dist/api/human-review.js +236 -0
- package/dist/api/nondeterminism.d.ts +300 -0
- package/dist/api/nondeterminism.js +250 -0
- package/dist/api/quality-metrics.d.ts +303 -0
- package/dist/api/quality-metrics.js +198 -0
- package/dist/api/roi-analytics.d.ts +263 -0
- package/dist/api/roi-analytics.js +204 -0
- package/dist/api/transcript-patterns.d.ts +204 -0
- package/dist/api/transcript-patterns.js +227 -0
- package/dist/core/client.d.ts +82 -8
- package/dist/core/client.js +223 -32
- package/dist/core/config.d.ts +1 -1
- package/dist/core/config.js +2 -2
- package/dist/core/types.d.ts +27 -2
- package/dist/core/types.js +1 -1
- package/dist/index.d.ts +415 -62
- package/dist/index.js +253 -37
- package/package.json +8 -4
package/dist/core/client.js
CHANGED
|
@@ -5,21 +5,46 @@
|
|
|
5
5
|
* Centralized HTTP client with authentication and error handling
|
|
6
6
|
*/
|
|
7
7
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
-
exports.ThinkHiveValidationError = exports.ThinkHiveApiError = void 0;
|
|
8
|
+
exports.IpWhitelistError = exports.RateLimitError = exports.AgentScopeError = exports.PermissionDeniedError = exports.ThinkHiveValidationError = exports.ThinkHiveApiError = exports.ThinkHiveError = void 0;
|
|
9
9
|
exports.apiRequest = apiRequest;
|
|
10
10
|
exports.apiRequestWithData = apiRequestWithData;
|
|
11
11
|
const config_1 = require("./config");
|
|
12
|
+
// ============================================================================
|
|
13
|
+
// RETRY CONFIGURATION
|
|
14
|
+
// ============================================================================
|
|
15
|
+
const MAX_RETRIES = 3;
|
|
16
|
+
const INITIAL_BACKOFF = 500; // ms
|
|
17
|
+
const MAX_BACKOFF = 8000; // ms
|
|
18
|
+
const RETRYABLE_CODES = new Set([408, 429, 500, 502, 503, 504]);
|
|
19
|
+
/**
|
|
20
|
+
* Calculate backoff time for retry attempt
|
|
21
|
+
*/
|
|
22
|
+
function calculateBackoff(attempt, retryAfter) {
|
|
23
|
+
if (retryAfter !== undefined) {
|
|
24
|
+
return Math.min(retryAfter * 1000, MAX_BACKOFF); // Convert seconds to ms
|
|
25
|
+
}
|
|
26
|
+
const backoff = INITIAL_BACKOFF * Math.pow(2, attempt);
|
|
27
|
+
return Math.min(backoff, MAX_BACKOFF);
|
|
28
|
+
}
|
|
12
29
|
/**
|
|
13
|
-
*
|
|
30
|
+
* Sleep for a given number of milliseconds
|
|
31
|
+
*/
|
|
32
|
+
function sleep(ms) {
|
|
33
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Make an authenticated API request with retry logic
|
|
14
37
|
*/
|
|
15
38
|
async function apiRequest(path, options = {}) {
|
|
16
39
|
const config = (0, config_1.getConfig)();
|
|
17
|
-
const { method = 'GET', body, headers = {}, apiVersion } = options;
|
|
40
|
+
const { method = 'GET', body, headers = {}, apiVersion, maxRetries, timeout } = options;
|
|
18
41
|
const version = apiVersion || config.apiVersion;
|
|
19
42
|
const url = `${config.endpoint}/api/${version}${path}`;
|
|
43
|
+
const retries = maxRetries ?? MAX_RETRIES;
|
|
44
|
+
const requestTimeout = timeout ?? 30000; // Default 30s
|
|
20
45
|
const requestHeaders = {
|
|
21
46
|
'Content-Type': 'application/json',
|
|
22
|
-
'X-SDK-Version': '3.
|
|
47
|
+
'X-SDK-Version': '3.2.0',
|
|
23
48
|
...headers,
|
|
24
49
|
};
|
|
25
50
|
if (config.apiKey) {
|
|
@@ -29,32 +54,108 @@ async function apiRequest(path, options = {}) {
|
|
|
29
54
|
requestHeaders['X-Agent-ID'] = config.agentId;
|
|
30
55
|
}
|
|
31
56
|
(0, config_1.debugLog)(`${method} ${url}`);
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
headers: requestHeaders,
|
|
35
|
-
body: body ? JSON.stringify(body) : undefined,
|
|
36
|
-
});
|
|
37
|
-
// Handle deprecation warnings
|
|
38
|
-
if (response.headers.has('Deprecation')) {
|
|
39
|
-
const sunset = response.headers.get('Sunset');
|
|
40
|
-
const link = response.headers.get('Link');
|
|
41
|
-
console.warn(`[ThinkHive] Deprecation warning: This endpoint is deprecated.` +
|
|
42
|
-
(sunset ? ` Sunset: ${sunset}` : '') +
|
|
43
|
-
(link ? ` Successor: ${link}` : ''));
|
|
44
|
-
}
|
|
45
|
-
if (!response.ok) {
|
|
46
|
-
const errorText = await response.text();
|
|
47
|
-
let errorData = {};
|
|
57
|
+
let lastError = null;
|
|
58
|
+
for (let attempt = 0; attempt <= retries; attempt++) {
|
|
48
59
|
try {
|
|
49
|
-
|
|
60
|
+
// Create abort controller for timeout
|
|
61
|
+
const controller = new AbortController();
|
|
62
|
+
const timeoutId = setTimeout(() => controller.abort(), requestTimeout);
|
|
63
|
+
let response;
|
|
64
|
+
try {
|
|
65
|
+
response = await fetch(url, {
|
|
66
|
+
method,
|
|
67
|
+
headers: requestHeaders,
|
|
68
|
+
body: body ? JSON.stringify(body) : undefined,
|
|
69
|
+
signal: controller.signal,
|
|
70
|
+
});
|
|
71
|
+
}
|
|
72
|
+
finally {
|
|
73
|
+
clearTimeout(timeoutId);
|
|
74
|
+
}
|
|
75
|
+
// Handle deprecation warnings
|
|
76
|
+
if (response.headers.has('Deprecation')) {
|
|
77
|
+
const sunset = response.headers.get('Sunset');
|
|
78
|
+
const link = response.headers.get('Link');
|
|
79
|
+
console.warn(`[ThinkHive] Deprecation warning: This endpoint is deprecated.` +
|
|
80
|
+
(sunset ? ` Sunset: ${sunset}` : '') +
|
|
81
|
+
(link ? ` Successor: ${link}` : ''));
|
|
82
|
+
}
|
|
83
|
+
// Check if response is retryable
|
|
84
|
+
if (RETRYABLE_CODES.has(response.status) && attempt < retries) {
|
|
85
|
+
// Get Retry-After header if present (for 429 responses)
|
|
86
|
+
let retryAfter;
|
|
87
|
+
if (response.status === 429) {
|
|
88
|
+
const retryAfterHeader = response.headers.get('Retry-After');
|
|
89
|
+
if (retryAfterHeader) {
|
|
90
|
+
retryAfter = parseFloat(retryAfterHeader);
|
|
91
|
+
if (isNaN(retryAfter)) {
|
|
92
|
+
retryAfter = undefined;
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
const backoff = calculateBackoff(attempt, retryAfter);
|
|
97
|
+
(0, config_1.debugLog)(`Retrying after ${backoff}ms (attempt ${attempt + 1}/${retries})`);
|
|
98
|
+
await sleep(backoff);
|
|
99
|
+
continue;
|
|
100
|
+
}
|
|
101
|
+
if (!response.ok) {
|
|
102
|
+
const errorText = await response.text();
|
|
103
|
+
let errorData = {};
|
|
104
|
+
try {
|
|
105
|
+
errorData = JSON.parse(errorText);
|
|
106
|
+
}
|
|
107
|
+
catch {
|
|
108
|
+
errorData = { message: errorText };
|
|
109
|
+
}
|
|
110
|
+
// Handle rate limit specifically
|
|
111
|
+
if (response.status === 429) {
|
|
112
|
+
const retryAfterHeader = response.headers.get('Retry-After');
|
|
113
|
+
const retryAfter = retryAfterHeader ? parseFloat(retryAfterHeader) * 1000 : 60000;
|
|
114
|
+
throw new RateLimitError(retryAfter);
|
|
115
|
+
}
|
|
116
|
+
throw new ThinkHiveApiError(errorData.message || `HTTP ${response.status}`, response.status, errorData.code);
|
|
117
|
+
}
|
|
118
|
+
return response.json();
|
|
50
119
|
}
|
|
51
|
-
catch {
|
|
52
|
-
|
|
120
|
+
catch (error) {
|
|
121
|
+
// Handle abort (timeout)
|
|
122
|
+
if (error instanceof Error && error.name === 'AbortError') {
|
|
123
|
+
lastError = new ThinkHiveApiError(`Request timed out after ${requestTimeout}ms`, 408);
|
|
124
|
+
if (attempt < retries) {
|
|
125
|
+
const backoff = calculateBackoff(attempt);
|
|
126
|
+
(0, config_1.debugLog)(`Request timed out, retrying after ${backoff}ms (attempt ${attempt + 1}/${retries})`);
|
|
127
|
+
await sleep(backoff);
|
|
128
|
+
continue;
|
|
129
|
+
}
|
|
130
|
+
throw lastError;
|
|
131
|
+
}
|
|
132
|
+
// Handle network errors (TypeError from fetch)
|
|
133
|
+
if (error instanceof TypeError) {
|
|
134
|
+
lastError = new ThinkHiveApiError(`Network error: ${error.message}`, 503);
|
|
135
|
+
if (attempt < retries) {
|
|
136
|
+
const backoff = calculateBackoff(attempt);
|
|
137
|
+
(0, config_1.debugLog)(`Network error, retrying after ${backoff}ms (attempt ${attempt + 1}/${retries})`);
|
|
138
|
+
await sleep(backoff);
|
|
139
|
+
continue;
|
|
140
|
+
}
|
|
141
|
+
throw lastError;
|
|
142
|
+
}
|
|
143
|
+
// Re-throw ThinkHive errors (non-retryable)
|
|
144
|
+
if (error instanceof ThinkHiveError) {
|
|
145
|
+
throw error;
|
|
146
|
+
}
|
|
147
|
+
// Unknown error
|
|
148
|
+
lastError = error instanceof Error ? error : new Error(String(error));
|
|
149
|
+
if (attempt < retries) {
|
|
150
|
+
const backoff = calculateBackoff(attempt);
|
|
151
|
+
await sleep(backoff);
|
|
152
|
+
continue;
|
|
153
|
+
}
|
|
154
|
+
throw new ThinkHiveApiError(`Request failed: ${lastError.message}`, 500);
|
|
53
155
|
}
|
|
54
|
-
const error = new ThinkHiveApiError(errorData.message || `HTTP ${response.status}`, response.status, errorData.code);
|
|
55
|
-
throw error;
|
|
56
156
|
}
|
|
57
|
-
|
|
157
|
+
// Should not reach here, but just in case
|
|
158
|
+
throw lastError || new ThinkHiveApiError('Request failed after all retries', 500);
|
|
58
159
|
}
|
|
59
160
|
/**
|
|
60
161
|
* Make an API request and extract data from response wrapper
|
|
@@ -69,21 +170,111 @@ async function apiRequestWithData(path, options = {}) {
|
|
|
69
170
|
// ============================================================================
|
|
70
171
|
// ERRORS
|
|
71
172
|
// ============================================================================
|
|
72
|
-
|
|
73
|
-
|
|
173
|
+
/**
|
|
174
|
+
* Base error class for ThinkHive SDK errors
|
|
175
|
+
*/
|
|
176
|
+
class ThinkHiveError extends Error {
|
|
177
|
+
constructor(message, code, statusCode, details) {
|
|
74
178
|
super(message);
|
|
75
|
-
this.statusCode = statusCode;
|
|
76
179
|
this.code = code;
|
|
180
|
+
this.statusCode = statusCode;
|
|
181
|
+
this.details = details;
|
|
182
|
+
this.name = 'ThinkHiveError';
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
exports.ThinkHiveError = ThinkHiveError;
|
|
186
|
+
/**
|
|
187
|
+
* API request error - returned from ThinkHive server
|
|
188
|
+
*/
|
|
189
|
+
class ThinkHiveApiError extends ThinkHiveError {
|
|
190
|
+
constructor(message, statusCode, code) {
|
|
191
|
+
super(message, code || 'API_ERROR', statusCode);
|
|
77
192
|
this.name = 'ThinkHiveApiError';
|
|
78
193
|
}
|
|
79
194
|
}
|
|
80
195
|
exports.ThinkHiveApiError = ThinkHiveApiError;
|
|
81
|
-
|
|
196
|
+
/**
|
|
197
|
+
* Validation error - invalid input parameters
|
|
198
|
+
*/
|
|
199
|
+
class ThinkHiveValidationError extends ThinkHiveError {
|
|
82
200
|
constructor(message, field) {
|
|
83
|
-
super(message);
|
|
201
|
+
super(message, 'VALIDATION_ERROR', 400, field ? { field } : undefined);
|
|
84
202
|
this.field = field;
|
|
85
203
|
this.name = 'ThinkHiveValidationError';
|
|
86
204
|
}
|
|
87
205
|
}
|
|
88
206
|
exports.ThinkHiveValidationError = ThinkHiveValidationError;
|
|
89
|
-
|
|
207
|
+
/**
|
|
208
|
+
* Permission denied error - API key lacks required permissions
|
|
209
|
+
* @example
|
|
210
|
+
* ```typescript
|
|
211
|
+
* try {
|
|
212
|
+
* await thinkHive.traces.create(...);
|
|
213
|
+
* } catch (error) {
|
|
214
|
+
* if (error instanceof PermissionDeniedError) {
|
|
215
|
+
* console.log('API key needs write permission');
|
|
216
|
+
* }
|
|
217
|
+
* }
|
|
218
|
+
* ```
|
|
219
|
+
*/
|
|
220
|
+
class PermissionDeniedError extends ThinkHiveError {
|
|
221
|
+
constructor(message, details) {
|
|
222
|
+
super(message, 'PERMISSION_DENIED', 403, details);
|
|
223
|
+
this.name = 'PermissionDeniedError';
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
exports.PermissionDeniedError = PermissionDeniedError;
|
|
227
|
+
/**
|
|
228
|
+
* Agent scope violation error - API key not authorized for this agent
|
|
229
|
+
* @example
|
|
230
|
+
* ```typescript
|
|
231
|
+
* // Create key scoped to specific agents
|
|
232
|
+
* const key = await thinkHive.apiKeys.create({
|
|
233
|
+
* name: 'Agent A Key',
|
|
234
|
+
* allowedAgentIds: ['agent-a-id']
|
|
235
|
+
* });
|
|
236
|
+
*
|
|
237
|
+
* // Attempting to use for Agent B will throw AgentScopeError
|
|
238
|
+
* ```
|
|
239
|
+
*/
|
|
240
|
+
class AgentScopeError extends ThinkHiveError {
|
|
241
|
+
constructor(agentId, allowedAgents) {
|
|
242
|
+
super(`API key not authorized for agent ${agentId}`, 'AGENT_SCOPE_VIOLATION', 403, { agentId, allowedAgents });
|
|
243
|
+
this.name = 'AgentScopeError';
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
exports.AgentScopeError = AgentScopeError;
|
|
247
|
+
/**
|
|
248
|
+
* Rate limit exceeded error
|
|
249
|
+
* @example
|
|
250
|
+
* ```typescript
|
|
251
|
+
* try {
|
|
252
|
+
* await thinkHive.traces.create(...);
|
|
253
|
+
* } catch (error) {
|
|
254
|
+
* if (error instanceof RateLimitError) {
|
|
255
|
+
* // Wait and retry
|
|
256
|
+
* await sleep(error.retryAfter);
|
|
257
|
+
* await thinkHive.traces.create(...);
|
|
258
|
+
* }
|
|
259
|
+
* }
|
|
260
|
+
* ```
|
|
261
|
+
*/
|
|
262
|
+
class RateLimitError extends ThinkHiveError {
|
|
263
|
+
constructor(retryAfter) {
|
|
264
|
+
super(`Rate limit exceeded. Retry after ${retryAfter}ms`, 'RATE_LIMIT_EXCEEDED', 429, { retryAfter });
|
|
265
|
+
this.retryAfter = retryAfter;
|
|
266
|
+
this.name = 'RateLimitError';
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
exports.RateLimitError = RateLimitError;
|
|
270
|
+
/**
|
|
271
|
+
* IP whitelist violation error
|
|
272
|
+
*/
|
|
273
|
+
class IpWhitelistError extends ThinkHiveError {
|
|
274
|
+
constructor(clientIp) {
|
|
275
|
+
super('Request IP address is not authorized for this API key', 'IP_WHITELIST_VIOLATION', 403, { clientIp });
|
|
276
|
+
this.name = 'IpWhitelistError';
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
exports.IpWhitelistError = IpWhitelistError;
|
|
280
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client.js","sourceRoot":"","sources":["../../src/core/client.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ThinkHive SDK v3.0 - HTTP Client\n *\n * Centralized HTTP client with authentication and error handling\n */\n\nimport { getConfig, debugLog } from './config';\nimport type { ApiResponse } from './types';\n\n// ============================================================================\n// RETRY CONFIGURATION\n// ============================================================================\n\nconst MAX_RETRIES = 3;\nconst INITIAL_BACKOFF = 500; // ms\nconst MAX_BACKOFF = 8000; // ms\nconst RETRYABLE_CODES = new Set([408, 429, 500, 502, 503, 504]);\n\n/**\n * Calculate backoff time for retry attempt\n */\nfunction calculateBackoff(attempt: number, retryAfter?: number): number {\n  if (retryAfter !== undefined) {\n    return Math.min(retryAfter * 1000, MAX_BACKOFF); // Convert seconds to ms\n  }\n  const backoff = INITIAL_BACKOFF * Math.pow(2, attempt);\n  return Math.min(backoff, MAX_BACKOFF);\n}\n\n/**\n * Sleep for a given number of milliseconds\n */\nfunction sleep(ms: number): Promise<void> {\n  return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\n// ============================================================================\n// HTTP CLIENT\n// ============================================================================\n\nexport interface RequestOptions {\n  method?: 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE';\n  body?: unknown;\n  headers?: Record<string, string>;\n  /** Override API version for this request */\n  apiVersion?: 'v1' | 'v2' | 'v3';\n  /** Override max retries for this request */\n  maxRetries?: number;\n  /** Request timeout in milliseconds */\n  timeout?: number;\n}\n\n/**\n * Make an authenticated API request with retry logic\n */\nexport async function apiRequest<T>(\n  path: string,\n  options: RequestOptions = {}\n): Promise<T> {\n  const config = getConfig();\n  const { method = 'GET', body, headers = {}, apiVersion, maxRetries, timeout } = options;\n\n  const version = apiVersion || config.apiVersion;\n  const url = `${config.endpoint}/api/${version}${path}`;\n  const retries = maxRetries ?? MAX_RETRIES;\n  const requestTimeout = timeout ?? 30000; // Default 30s\n\n  const requestHeaders: Record<string, string> = {\n    'Content-Type': 'application/json',\n    'X-SDK-Version': '3.2.0',\n    ...headers,\n  };\n\n  if (config.apiKey) {\n    requestHeaders['Authorization'] = `Bearer ${config.apiKey}`;\n  } else if (config.agentId) {\n    requestHeaders['X-Agent-ID'] = config.agentId;\n  }\n\n  debugLog(`${method} ${url}`);\n\n  let lastError: Error | null = null;\n\n  for (let attempt = 0; attempt <= retries; attempt++) {\n    try {\n      // Create abort controller for timeout\n      const controller = new AbortController();\n      const timeoutId = setTimeout(() => controller.abort(), requestTimeout);\n\n      let response: Response;\n      try {\n        response = await fetch(url, {\n          method,\n          headers: requestHeaders,\n          body: body ? JSON.stringify(body) : undefined,\n          signal: controller.signal,\n        });\n      } finally {\n        clearTimeout(timeoutId);\n      }\n\n      // Handle deprecation warnings\n      if (response.headers.has('Deprecation')) {\n        const sunset = response.headers.get('Sunset');\n        const link = response.headers.get('Link');\n        console.warn(\n          `[ThinkHive] Deprecation warning: This endpoint is deprecated.` +\n            (sunset ? ` Sunset: ${sunset}` : '') +\n            (link ? ` Successor: ${link}` : '')\n        );\n      }\n\n      // Check if response is retryable\n      if (RETRYABLE_CODES.has(response.status) && attempt < retries) {\n        // Get Retry-After header if present (for 429 responses)\n        let retryAfter: number | undefined;\n        if (response.status === 429) {\n          const retryAfterHeader = response.headers.get('Retry-After');\n          if (retryAfterHeader) {\n            retryAfter = parseFloat(retryAfterHeader);\n            if (isNaN(retryAfter)) {\n              retryAfter = undefined;\n            }\n          }\n        }\n\n        const backoff = calculateBackoff(attempt, retryAfter);\n        debugLog(`Retrying after ${backoff}ms (attempt ${attempt + 1}/${retries})`);\n        await sleep(backoff);\n        continue;\n      }\n\n      if (!response.ok) {\n        const errorText = await response.text();\n        let errorData: { message?: string; code?: string } = {};\n        try {\n          errorData = JSON.parse(errorText);\n        } catch {\n          errorData = { message: errorText };\n        }\n\n        // Handle rate limit specifically\n        if (response.status === 429) {\n          const retryAfterHeader = response.headers.get('Retry-After');\n          const retryAfter = retryAfterHeader ? parseFloat(retryAfterHeader) * 1000 : 60000;\n          throw new RateLimitError(retryAfter);\n        }\n\n        throw new ThinkHiveApiError(\n          errorData.message || `HTTP ${response.status}`,\n          response.status,\n          errorData.code\n        );\n      }\n\n      return response.json() as Promise<T>;\n    } catch (error) {\n      // Handle abort (timeout)\n      if (error instanceof Error && error.name === 'AbortError') {\n        lastError = new ThinkHiveApiError(`Request timed out after ${requestTimeout}ms`, 408);\n        if (attempt < retries) {\n          const backoff = calculateBackoff(attempt);\n          debugLog(`Request timed out, retrying after ${backoff}ms (attempt ${attempt + 1}/${retries})`);\n          await sleep(backoff);\n          continue;\n        }\n        throw lastError;\n      }\n\n      // Handle network errors (TypeError from fetch)\n      if (error instanceof TypeError) {\n        lastError = new ThinkHiveApiError(`Network error: ${error.message}`, 503);\n        if (attempt < retries) {\n          const backoff = calculateBackoff(attempt);\n          debugLog(`Network error, retrying after ${backoff}ms (attempt ${attempt + 1}/${retries})`);\n          await sleep(backoff);\n          continue;\n        }\n        throw lastError;\n      }\n\n      // Re-throw ThinkHive errors (non-retryable)\n      if (error instanceof ThinkHiveError) {\n        throw error;\n      }\n\n      // Unknown error\n      lastError = error instanceof Error ? error : new Error(String(error));\n      if (attempt < retries) {\n        const backoff = calculateBackoff(attempt);\n        await sleep(backoff);\n        continue;\n      }\n      throw new ThinkHiveApiError(`Request failed: ${lastError.message}`, 500);\n    }\n  }\n\n  // Should not reach here, but just in case\n  throw lastError || new ThinkHiveApiError('Request failed after all retries', 500);\n}\n\n/**\n * Make an API request and extract data from response wrapper\n */\nexport async function apiRequestWithData<T>(\n  path: string,\n  options: RequestOptions = {}\n): Promise<T> {\n  const response = await apiRequest<ApiResponse<T>>(path, options);\n  if (!response.success) {\n    throw new ThinkHiveApiError(\n      response.error?.message || 'Unknown error',\n      500,\n      response.error?.code\n    );\n  }\n  return response.data as T;\n}\n\n// ============================================================================\n// ERRORS\n// ============================================================================\n\n/**\n * Base error class for ThinkHive SDK errors\n */\nexport class ThinkHiveError extends Error {\n  constructor(\n    message: string,\n    public readonly code: string,\n    public readonly statusCode?: number,\n    public readonly details?: Record<string, unknown>\n  ) {\n    super(message);\n    this.name = 'ThinkHiveError';\n  }\n}\n\n/**\n * API request error - returned from ThinkHive server\n */\nexport class ThinkHiveApiError extends ThinkHiveError {\n  constructor(\n    message: string,\n    statusCode: number,\n    code?: string\n  ) {\n    super(message, code || 'API_ERROR', statusCode);\n    this.name = 'ThinkHiveApiError';\n  }\n}\n\n/**\n * Validation error - invalid input parameters\n */\nexport class ThinkHiveValidationError extends ThinkHiveError {\n  constructor(\n    message: string,\n    public readonly field?: string\n  ) {\n    super(message, 'VALIDATION_ERROR', 400, field ? { field } : undefined);\n    this.name = 'ThinkHiveValidationError';\n  }\n}\n\n/**\n * Permission denied error - API key lacks required permissions\n * @example\n * ```typescript\n * try {\n *   await thinkHive.traces.create(...);\n * } catch (error) {\n *   if (error instanceof PermissionDeniedError) {\n *     console.log('API key needs write permission');\n *   }\n * }\n * ```\n */\nexport class PermissionDeniedError extends ThinkHiveError {\n  constructor(message: string, details?: Record<string, unknown>) {\n    super(message, 'PERMISSION_DENIED', 403, details);\n    this.name = 'PermissionDeniedError';\n  }\n}\n\n/**\n * Agent scope violation error - API key not authorized for this agent\n * @example\n * ```typescript\n * // Create key scoped to specific agents\n * const key = await thinkHive.apiKeys.create({\n *   name: 'Agent A Key',\n *   allowedAgentIds: ['agent-a-id']\n * });\n *\n * // Attempting to use for Agent B will throw AgentScopeError\n * ```\n */\nexport class AgentScopeError extends ThinkHiveError {\n  constructor(\n    agentId: string,\n    allowedAgents: string[]\n  ) {\n    super(\n      `API key not authorized for agent ${agentId}`,\n      'AGENT_SCOPE_VIOLATION',\n      403,\n      { agentId, allowedAgents }\n    );\n    this.name = 'AgentScopeError';\n  }\n}\n\n/**\n * Rate limit exceeded error\n * @example\n * ```typescript\n * try {\n *   await thinkHive.traces.create(...);\n * } catch (error) {\n *   if (error instanceof RateLimitError) {\n *     // Wait and retry\n *     await sleep(error.retryAfter);\n *     await thinkHive.traces.create(...);\n *   }\n * }\n * ```\n */\nexport class RateLimitError extends ThinkHiveError {\n  constructor(\n    public readonly retryAfter: number\n  ) {\n    super(\n      `Rate limit exceeded. Retry after ${retryAfter}ms`,\n      'RATE_LIMIT_EXCEEDED',\n      429,\n      { retryAfter }\n    );\n    this.name = 'RateLimitError';\n  }\n}\n\n/**\n * IP whitelist violation error\n */\nexport class IpWhitelistError extends ThinkHiveError {\n  constructor(clientIp: string) {\n    super(\n      'Request IP address is not authorized for this API key',\n      'IP_WHITELIST_VIOLATION',\n      403,\n      { clientIp }\n    );\n    this.name = 'IpWhitelistError';\n  }\n}\n"]}
|
package/dist/core/config.d.ts
CHANGED
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
* ThinkHive SDK v3.0 - Configuration
|
|
3
3
|
*/
|
|
4
4
|
import type { InitOptions, Framework } from './types';
|
|
5
|
-
export declare const SDK_VERSION = "3.
|
|
5
|
+
export declare const SDK_VERSION = "3.3.0";
|
|
6
6
|
export declare const DEFAULT_ENDPOINT = "https://thinkhivemind-h25z7pvd3q-uc.a.run.app";
|
|
7
7
|
export declare const DEFAULT_SERVICE_NAME = "my-ai-agent";
|
|
8
8
|
export interface ResolvedConfig {
|
package/dist/core/config.js
CHANGED
|
@@ -12,7 +12,7 @@ exports.debugLog = debugLog;
|
|
|
12
12
|
// ============================================================================
|
|
13
13
|
// CONSTANTS
|
|
14
14
|
// ============================================================================
|
|
15
|
-
exports.SDK_VERSION = '3.
|
|
15
|
+
exports.SDK_VERSION = '3.3.0';
|
|
16
16
|
exports.DEFAULT_ENDPOINT = 'https://thinkhivemind-h25z7pvd3q-uc.a.run.app';
|
|
17
17
|
exports.DEFAULT_SERVICE_NAME = 'my-ai-agent';
|
|
18
18
|
let config = null;
|
|
@@ -73,4 +73,4 @@ function debugLog(...args) {
|
|
|
73
73
|
console.log('[ThinkHive]', ...args);
|
|
74
74
|
}
|
|
75
75
|
}
|
|
76
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
76
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvcmUvY29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7R0FFRzs7O0FBcUNILGdDQXVCQztBQU1ELDhCQUtDO0FBS0Qsc0NBRUM7QUFLRCxrQ0FHQztBQUtELDRCQUlDO0FBM0ZELCtFQUErRTtBQUMvRSxZQUFZO0FBQ1osK0VBQStFO0FBRWxFLFFBQUEsV0FBVyxHQUFHLE9BQU8sQ0FBQztBQUN0QixRQUFBLGdCQUFnQixHQUFHLCtDQUErQyxDQUFDO0FBQ25FLFFBQUEsb0JBQW9CLEdBQUcsYUFBYSxDQUFDO0FBaUJsRCxJQUFJLE1BQU0sR0FBMEIsSUFBSSxDQUFDO0FBQ3pDLElBQUksV0FBVyxHQUFHLEtBQUssQ0FBQztBQUV4QiwrRUFBK0U7QUFDL0UsbUJBQW1CO0FBQ25CLCtFQUErRTtBQUUvRTs7R0FFRztBQUNILFNBQWdCLFVBQVUsQ0FBQyxVQUF1QixFQUFFO0lBQ2xELE1BQU0sTUFBTSxHQUFHLE9BQU8sQ0FBQyxNQUFNLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxpQkFBaUIsSUFBSSxFQUFFLENBQUM7SUFDckUsTUFBTSxPQUFPLEdBQUcsT0FBTyxDQUFDLE9BQU8sSUFBSSxPQUFPLENBQUMsR0FBRyxDQUFDLGtCQUFrQixJQUFJLEVBQUUsQ0FBQztJQUV4RSxJQUFJLENBQUMsTUFBTSxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDeEIsTUFBTSxJQUFJLEtBQUssQ0FDYiw4RUFBOEUsQ0FDL0UsQ0FBQztJQUNKLENBQUM7SUFFRCxNQUFNLEdBQUc7UUFDUCxNQUFNO1FBQ04sT0FBTztRQUNQLFFBQVEsRUFBRSxPQUFPLENBQUMsUUFBUSxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsa0JBQWtCLElBQUksd0JBQWdCO1FBQ2hGLFdBQVcsRUFBRSxPQUFPLENBQUMsV0FBVyxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsc0JBQXNCLElBQUksNEJBQW9CO1FBQzlGLGNBQWMsRUFBRSxPQUFPLENBQUMsY0FBYyxJQUFJLEtBQUs7UUFDL0MsVUFBVSxFQUFFLE9BQU8sQ0FBQyxVQUFVLElBQUksQ0FBQyxXQUFXLEVBQUUsUUFBUSxDQUFDO1FBQ3pELEtBQUssRUFBRSxPQUFPLENBQUMsS0FBSyxJQUFJLEtBQUs7UUFDN0IsVUFBVSxFQUFFLE9BQU8sQ0FBQyxVQUFVLElBQUksSUFBSTtLQUN2QyxDQUFDO0lBRUYsV0FBVyxHQUFHLElBQUksQ0FBQztJQUNuQixPQUFPLE1BQU0sQ0FBQztBQUNoQixDQUFDO0FBRUQ7OztHQUdHO0FBQ0gsU0FBZ0IsU0FBUztJQUN2QixJQUFJLENBQUMsTUFBTSxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDNUIsTUFBTSxJQUFJLEtBQUssQ0FBQyxtREFBbUQsQ0FBQyxDQUFDO0lBQ3ZFLENBQUM7SUFDRCxPQUFPLE1BQU0sQ0FBQztBQUNoQixDQUFDO0FBRUQ7O0dBRUc7QUFDSCxTQUFnQixhQUFhO0lBQzNCLE9BQU8sV0FBVyxDQUFDO0FBQ3JCLENBQUM7QUFFRDs7R0FFRztBQUNILFNBQWdCLFdBQVc7SUFDekIsTUFBTSxHQUFHLElBQUksQ0FBQztJQUNkLFdBQVcsR0FBRyxLQUFLLENBQUM7QUFDdEIsQ0FBQztBQUVEOztHQUVHO0FBQ0gsU0FBZ0IsUUFBUSxDQUFDLEdBQUcsSUFBZTtJQUN6QyxJQUFJLE1BQU0sRUFBRSxLQUFLLEVBQUUsQ0FBQztRQUNsQixPQUFPLENBQUMsR0FBRyxDQUFDLGFBQWEsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDO0lBQ3RDLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBUaGlua0hpdmUgU0RLIHYzLjAgLSBDb25maWd1cmF0aW9uXG4gKi9cblxuaW1wb3J0IHR5cGUgeyBJbml0T3B0aW9ucywgRnJhbWV3b3JrIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIENPTlNUQU5UU1xuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG5leHBvcnQgY29uc3QgU0RLX1ZFUlNJT04gPSAnMy4zLjAnO1xuZXhwb3J0IGNvbnN0IERFRkFVTFRfRU5EUE9JTlQgPSAnaHR0cHM6Ly90aGlua2hpdmVtaW5kLWgyNXo3cHZkM3EtdWMuYS5ydW4uYXBwJztcbmV4cG9ydCBjb25zdCBERUZBVUxUX1NFUlZJQ0VfTkFNRSA9ICdteS1haS1hZ2VudCc7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIEdMT0JBTCBDT05GSUcgU1RBVEVcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0IGludGVyZmFjZSBSZXNvbHZlZENvbmZpZyB7XG4gIGFwaUtleTogc3RyaW5nO1xuICBhZ2VudElkOiBzdHJpbmc7XG4gIGVuZHBvaW50OiBzdHJpbmc7XG4gIHNlcnZpY2VOYW1lOiBzdHJpbmc7XG4gIGF1dG9JbnN0cnVtZW50OiBib29sZWFuO1xuICBmcmFtZXdvcmtzOiBGcmFtZXdvcmtbXTtcbiAgZGVidWc6IGJvb2xlYW47XG4gIGFwaVZlcnNpb246ICd2MScgfCAndjMnO1xufVxuXG5sZXQgY29uZmlnOiBSZXNvbHZlZENvbmZpZyB8IG51bGwgPSBudWxsO1xubGV0IGluaXRpYWxpemVkID0gZmFsc2U7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIENPTkZJRyBGVU5DVElPTlNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBJbml0aWFsaXplIGNvbmZpZ3VyYXRpb24gZnJvbSBvcHRpb25zXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpbml0Q29uZmlnKG9wdGlvbnM6IEluaXRPcHRpb25zID0ge30pOiBSZXNvbHZlZENvbmZpZyB7XG4gIGNvbnN0IGFwaUtleSA9IG9wdGlvbnMuYXBpS2V5IHx8IHByb2Nlc3MuZW52LlRISU5LSElWRV9BUElfS0VZIHx8ICcnO1xuICBjb25zdCBhZ2VudElkID0gb3B0aW9ucy5hZ2VudElkIHx8IHByb2Nlc3MuZW52LlRISU5LSElWRV9BR0VOVF9JRCB8fCAnJztcblxuICBpZiAoIWFwaUtleSAmJiAhYWdlbnRJZCkge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICdFaXRoZXIgYXBpS2V5IG9yIGFnZW50SWQgbXVzdCBiZSBwcm92aWRlZCAob3Igc2V0IFRISU5LSElWRV9BUElfS0VZIGVudiB2YXIpJ1xuICAgICk7XG4gIH1cblxuICBjb25maWcgPSB7XG4gICAgYXBpS2V5LFxuICAgIGFnZW50SWQsXG4gICAgZW5kcG9pbnQ6IG9wdGlvbnMuZW5kcG9pbnQgfHwgcHJvY2Vzcy5lbnYuVEhJTktISVZFX0VORFBPSU5UIHx8IERFRkFVTFRfRU5EUE9JTlQsXG4gICAgc2VydmljZU5hbWU6IG9wdGlvbnMuc2VydmljZU5hbWUgfHwgcHJvY2Vzcy5lbnYuVEhJTktISVZFX1NFUlZJQ0VfTkFNRSB8fCBERUZBVUxUX1NFUlZJQ0VfTkFNRSxcbiAgICBhdXRvSW5zdHJ1bWVudDogb3B0aW9ucy5hdXRvSW5zdHJ1bWVudCA/PyBmYWxzZSxcbiAgICBmcmFtZXdvcmtzOiBvcHRpb25zLmZyYW1ld29ya3MgfHwgWydsYW5nY2hhaW4nLCAnb3BlbmFpJ10sXG4gICAgZGVidWc6IG9wdGlvbnMuZGVidWcgPz8gZmFsc2UsXG4gICAgYXBpVmVyc2lvbjogb3B0aW9ucy5hcGlWZXJzaW9uID8/ICd2MycsXG4gIH07XG5cbiAgaW5pdGlhbGl6ZWQgPSB0cnVlO1xuICByZXR1cm4gY29uZmlnO1xufVxuXG4vKipcbiAqIEdldCBjdXJyZW50IGNvbmZpZ3VyYXRpb25cbiAqIEB0aHJvd3MgaWYgbm90IGluaXRpYWxpemVkXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRDb25maWcoKTogUmVzb2x2ZWRDb25maWcge1xuICBpZiAoIWNvbmZpZyB8fCAhaW5pdGlhbGl6ZWQpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ1RoaW5rSGl2ZSBTREsgbm90IGluaXRpYWxpemVkLiBDYWxsIGluaXQoKSBmaXJzdC4nKTtcbiAgfVxuICByZXR1cm4gY29uZmlnO1xufVxuXG4vKipcbiAqIENoZWNrIGlmIFNESyBpcyBpbml0aWFsaXplZFxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNJbml0aWFsaXplZCgpOiBib29sZWFuIHtcbiAgcmV0dXJuIGluaXRpYWxpemVkO1xufVxuXG4vKipcbiAqIFJlc2V0IGNvbmZpZ3VyYXRpb24gKGZvciB0ZXN0aW5nKVxuICovXG5leHBvcnQgZnVuY3Rpb24gcmVzZXRDb25maWcoKTogdm9pZCB7XG4gIGNvbmZpZyA9IG51bGw7XG4gIGluaXRpYWxpemVkID0gZmFsc2U7XG59XG5cbi8qKlxuICogRGVidWcgbG9nIGhlbHBlclxuICovXG5leHBvcnQgZnVuY3Rpb24gZGVidWdMb2coLi4uYXJnczogdW5rbm93bltdKTogdm9pZCB7XG4gIGlmIChjb25maWc/LmRlYnVnKSB7XG4gICAgY29uc29sZS5sb2coJ1tUaGlua0hpdmVdJywgLi4uYXJncyk7XG4gIH1cbn1cbiJdfQ==
|
package/dist/core/types.d.ts
CHANGED
|
@@ -50,6 +50,15 @@ export interface RunOptions {
|
|
|
50
50
|
ticketLinking?: TicketLinkingOptions;
|
|
51
51
|
/** Custom metadata */
|
|
52
52
|
metadata?: Record<string, unknown>;
|
|
53
|
+
/**
|
|
54
|
+
* Request evaluation when run is ingested.
|
|
55
|
+
* Overrides agent's auto_evaluate setting for this run.
|
|
56
|
+
*
|
|
57
|
+
* - true: Force evaluation on this run
|
|
58
|
+
* - false: Skip evaluation even if agent has auto_evaluate enabled
|
|
59
|
+
* - undefined: Use agent's auto_evaluate setting (default)
|
|
60
|
+
*/
|
|
61
|
+
runEvaluation?: boolean;
|
|
53
62
|
}
|
|
54
63
|
export type RunOutcome = 'resolved' | 'escalated' | 'abandoned' | 'failed' | 'transferred' | 'pending';
|
|
55
64
|
export interface ConversationMessage {
|
|
@@ -93,6 +102,11 @@ export interface TicketLinkingOptions {
|
|
|
93
102
|
customFieldValue?: string;
|
|
94
103
|
}
|
|
95
104
|
export type LinkMethod = 'sdk_explicit' | 'zendesk_marker' | 'custom_field' | 'middleware_stamp' | 'session_match' | 'email_time_window' | 'manual';
|
|
105
|
+
/**
|
|
106
|
+
* Classification flags for categorizing trace issues
|
|
107
|
+
* Separate from binary outcome (success/failure)
|
|
108
|
+
*/
|
|
109
|
+
export type TraceCustomFlag = 'hallucination' | 'policy_violation' | 'tone_issue' | 'retrieval_miss' | 'error';
|
|
96
110
|
/**
|
|
97
111
|
* Trace options (v2 format, converted to runs internally)
|
|
98
112
|
* @deprecated Use RunOptions instead
|
|
@@ -104,8 +118,10 @@ export interface TraceOptions {
|
|
|
104
118
|
agentResponse: string;
|
|
105
119
|
/** User's detected intent */
|
|
106
120
|
userIntent?: string;
|
|
107
|
-
/**
|
|
108
|
-
outcome?: 'success' | 'failure'
|
|
121
|
+
/** Binary outcome: success or failure */
|
|
122
|
+
outcome?: 'success' | 'failure';
|
|
123
|
+
/** Classification flags for categorizing the issue type (hallucination, policy_violation, etc.) */
|
|
124
|
+
customFlags?: TraceCustomFlag[];
|
|
109
125
|
/** Duration in milliseconds */
|
|
110
126
|
duration?: number;
|
|
111
127
|
/** Session ID for conversation tracking */
|
|
@@ -121,6 +137,15 @@ export interface TraceOptions {
|
|
|
121
137
|
businessContext?: BusinessContext;
|
|
122
138
|
/** Custom metadata */
|
|
123
139
|
metadata?: Record<string, unknown>;
|
|
140
|
+
/**
|
|
141
|
+
* Request evaluation when trace is ingested.
|
|
142
|
+
* Overrides agent's auto_evaluate setting for this trace.
|
|
143
|
+
*
|
|
144
|
+
* - true: Force evaluation on this trace
|
|
145
|
+
* - false: Skip evaluation even if agent has auto_evaluate enabled
|
|
146
|
+
* - undefined: Use agent's auto_evaluate setting (default)
|
|
147
|
+
*/
|
|
148
|
+
runEvaluation?: boolean;
|
|
124
149
|
}
|
|
125
150
|
export interface SpanData {
|
|
126
151
|
id?: string;
|
package/dist/core/types.js
CHANGED
|
@@ -5,4 +5,4 @@
|
|
|
5
5
|
* Run-centric types with facts vs inferences and customer context snapshots
|
|
6
6
|
*/
|
|
7
7
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/core/types.ts"],"names":[],"mappings":";AAAA;;;;GAIG","sourcesContent":["/**\n * ThinkHive SDK v3.0 - Core Types\n *\n * Run-centric types with facts vs inferences and customer context snapshots\n */\n\n// ============================================================================\n// INIT OPTIONS\n// ============================================================================\n\nexport interface InitOptions {\n  /** ThinkHive API key (starts with th_) */\n  apiKey?: string;\n  /** Agent ID */\n  agentId?: string;\n  /** ThinkHive API endpoint */\n  endpoint?: string;\n  /** Service name for traces */\n  serviceName?: string;\n  /** Enable auto-instrumentation */\n  autoInstrument?: boolean;\n  /** Frameworks to auto-instrument */\n  frameworks?: Framework[];\n  /** Enable debug logging */\n  debug?: boolean;\n  /** API version (default: 'v3') */\n  apiVersion?: 'v1' | 'v3';\n}\n\nexport type Framework =\n  | 'langchain'\n  | 'langgraph'\n  | 'openai'\n  | 'openai-assistants'\n  | 'anthropic'\n  | 'llamaindex'\n  | 'n8n'\n  | 'voiceflow';\n\n// ============================================================================\n// RUN OPTIONS (v3)\n// ============================================================================\n\n/**\n * Options for creating a run (v3 atomic unit)\n */\nexport interface RunOptions {\n  /** Agent ID */\n  agentId: string;\n  /** When the run started */\n  startedAt?: string | Date;\n  /** When the run ended */\n  endedAt?: string | Date;\n  /** Run outcome */\n  outcome?: RunOutcome;\n  /** Reason for the outcome */\n  outcomeReason?: string;\n  /** Conversation messages */\n  conversationMessages: ConversationMessage[];\n  /** Model ID used */\n  modelId?: string;\n  /** Prompt version */\n  promptVersion?: string;\n  /** Session ID for conversation tracking */\n  sessionId?: string;\n  /** Customer context snapshot (ARR, health AS OF this run) */\n  customerContext?: CustomerContextSnapshot;\n  /** Ticket linking options */\n  ticketLinking?: TicketLinkingOptions;\n  /** Custom metadata */\n  metadata?: Record<string, unknown>;\n}\n\nexport type RunOutcome =\n  | 'resolved'\n  | 'escalated'\n  | 'abandoned'\n  | 'failed'\n  | 'transferred'\n  | 'pending';\n\nexport interface ConversationMessage {\n  role: 'user' | 'assistant' | 'system' | 'tool';\n  content: string;\n  timestamp?: string;\n  metadata?: Record<string, unknown>;\n}\n\n// ============================================================================\n// CUSTOMER CONTEXT SNAPSHOT (v3)\n// ============================================================================\n\n/**\n * Customer context captured AS OF the run time (not current values)\n */\nexport interface CustomerContextSnapshot {\n  /** Customer account ID */\n  customerId?: string;\n  /** ARR at the time of the run (not current ARR) */\n  arr?: number;\n  /** Health score at the time of the run (0-100) */\n  healthScore?: number;\n  /** Customer segment (e.g., 'enterprise', 'mid-market', 'smb') */\n  segment?: string;\n  /** When this snapshot was captured */\n  capturedAt?: string;\n  /** Custom fields (will be validated against schema) */\n  customFields?: Record<string, unknown>;\n}\n\n// ============================================================================\n// TICKET LINKING (v3)\n// ============================================================================\n\n/**\n * Options for deterministic ticket linking\n */\nexport interface TicketLinkingOptions {\n  /** Link method to use */\n  method: LinkMethod;\n  /** Ticket ID (for sdk_explicit method) */\n  ticketId?: string;\n  /** External ticket ID (e.g., Zendesk ticket ID) */\n  externalTicketId?: string;\n  /** Platform for external ticket */\n  platform?: 'zendesk' | 'intercom' | 'salesforce' | 'freshdesk';\n  /** Custom field name for linking */\n  customFieldName?: string;\n  /** Custom field value */\n  customFieldValue?: string;\n}\n\nexport type LinkMethod =\n  | 'sdk_explicit'      // Direct SDK call with ticket ID (confidence: 1.0)\n  | 'zendesk_marker'    // Embedded THID marker in response (confidence: 1.0)\n  | 'custom_field'      // Zendesk custom field (confidence: 1.0)\n  | 'middleware_stamp'  // Middleware-injected trace ID (confidence: 0.98)\n  | 'session_match'     // Session ID correlation (confidence: 0.95)\n  | 'email_time_window' // Email + 15min window (confidence: 0.6)\n  | 'manual';           // Human-assigned (confidence: 1.0)\n\n// ============================================================================\n// TRACE OPTIONS (v2 compatibility)\n// ============================================================================\n\n/**\n * Trace options (v2 format, converted to runs internally)\n * @deprecated Use RunOptions instead\n */\nexport interface TraceOptions {\n  /** User's message/query */\n  userMessage: string;\n  /** Agent's response */\n  agentResponse: string;\n  /** User's detected intent */\n  userIntent?: string;\n  /** Outcome: success, failure, partial_success */\n  outcome?: 'success' | 'failure' | 'partial_success';\n  /** Duration in milliseconds */\n  duration?: number;\n  /** Session ID for conversation tracking */\n  sessionId?: string;\n  /** Conversation history */\n  conversationHistory?: Array<{ role: string; content: string }>;\n  /** Span data for detailed analysis */\n  spans?: SpanData[];\n  /** Business context for ROI calculation */\n  businessContext?: BusinessContext;\n  /** Custom metadata */\n  metadata?: Record<string, unknown>;\n}\n\nexport interface SpanData {\n  id?: string;\n  name: string;\n  type: 'llm' | 'tool' | 'retrieval' | 'embedding' | 'chain' | 'custom';\n  startTime?: Date;\n  endTime?: Date;\n  durationMs?: number;\n  status?: 'ok' | 'error' | 'timeout';\n  error?: string;\n  input?: unknown;\n  output?: unknown;\n  // LLM-specific\n  model?: string;\n  provider?: string;\n  promptTokens?: number;\n  completionTokens?: number;\n  // Tool-specific\n  toolName?: string;\n  toolParameters?: Record<string, unknown>;\n  // Retrieval-specific\n  query?: string;\n  documentCount?: number;\n  topScore?: number;\n  sources?: string[];\n  // Children\n  children?: SpanData[];\n}\n\n/**\n * @deprecated Use CustomerContextSnapshot instead\n */\nexport interface BusinessContext {\n  /** Customer ID */\n  customerId?: string;\n  /** Transaction value in dollars */\n  transactionValue?: number;\n  /** Priority level */\n  priority?: 'low' | 'medium' | 'high' | 'critical';\n  /** Department */\n  department?: string;\n  /** Industry vertical */\n  industry?: string;\n  /** Custom fields */\n  custom?: Record<string, unknown>;\n}\n\n// ============================================================================\n// CLAIM TYPES (v3)\n// ============================================================================\n\n/**\n * A claim about a run (fact vs inference)\n */\nexport interface Claim {\n  id: string;\n  analysisId: string;\n  /** Type of claim */\n  claimType: ClaimType;\n  /** Category of the claim */\n  claimCategory: ClaimCategory;\n  /** The claim text */\n  claimText: string;\n  /** Confidence in this claim (0-1) */\n  confidence: number;\n  /** How well calibrated is this confidence */\n  confidenceCalibration: ConfidenceCalibration;\n  /** Evidence supporting this claim */\n  evidence: EvidenceReference[];\n  /** Is this claim explainable to end users */\n  isExplainable: boolean;\n  /** Probability value for probability claims */\n  probabilityValue?: number;\n  /** Human-assigned verification status */\n  humanVerified?: boolean;\n  /** Human verification verdict */\n  humanVerdict?: 'confirmed' | 'rejected' | 'modified';\n  createdAt?: string;\n}\n\nexport type ClaimType = 'observed' | 'inferred' | 'computed';\nexport type ClaimCategory =\n  | 'outcome'\n  | 'root_cause'\n  | 'customer_impact'\n  | 'churn_risk'\n  | 'revenue_impact'\n  | 'quality'\n  | 'other';\nexport type ConfidenceCalibration = 'calibrated' | 'uncalibrated' | 'needs_more_data';\n\nexport interface EvidenceReference {\n  /** Type of evidence */\n  type: 'span' | 'message' | 'tool_call' | 'external' | 'computed';\n  /** ID of the referenced item */\n  referenceId: string;\n  /** How relevant is this evidence */\n  relevance: string;\n  /** Confidence in this evidence */\n  confidence: number;\n}\n\n// ============================================================================\n// ANALYSIS RESULT (v3)\n// ============================================================================\n\nexport interface AnalysisResult {\n  id: string;\n  runId: string;\n  analysisVersion: string;\n  modelUsed: string;\n  /** Overall outcome verdict */\n  outcomeVerdict: 'success' | 'partial_success' | 'failure';\n  /** Confidence in the verdict (0-1) */\n  outcomeConfidence: number;\n  /** Root cause category if failure */\n  rootCauseCategory?: string;\n  /** Confidence in root cause (0-1) */\n  rootCauseConfidence?: number;\n  /** Is this the current analysis */\n  isCurrent: boolean;\n  /** ID of analysis that superseded this one */\n  supersededBy?: string;\n  /** Claims from this analysis */\n  claims: Claim[];\n  analyzedAt?: string;\n  createdAt?: string;\n}\n\n// ============================================================================\n// CALIBRATION TYPES (v3)\n// ============================================================================\n\nexport interface CalibrationStatus {\n  agentId: string;\n  predictionType: PredictionType;\n  /** Brier score (lower is better, <0.1 is good) */\n  brierScore: number;\n  /** Expected Calibration Error */\n  ece: number;\n  /** Sample count */\n  sampleCount: number;\n  /** Is the model well-calibrated */\n  isCalibrated: boolean;\n  /** Calibration by confidence bucket */\n  buckets: CalibrationBucket[];\n  lastUpdated?: string;\n}\n\nexport type PredictionType =\n  | 'outcome'\n  | 'churn_risk'\n  | 'escalation_risk'\n  | 'resolution_time'\n  | 'customer_satisfaction';\n\nexport interface CalibrationBucket {\n  bucketStart: number;\n  bucketEnd: number;\n  predictedProbability: number;\n  actualFrequency: number;\n  sampleCount: number;\n}\n\n// ============================================================================\n// ROI TYPES (v3)\n// ============================================================================\n\nexport interface RoiConfig {\n  /** Cost per support ticket deflected */\n  ticketDeflectionValue: number;\n  /** Cost per escalation */\n  escalationCost: number;\n  /** Cost per hour of agent time */\n  agentHourlyCost: number;\n  /** Average resolution time (minutes) */\n  avgResolutionMinutes: number;\n  /** CSAT impact multiplier */\n  csatImpactMultiplier: number;\n}\n\nexport interface RoiSummary {\n  period: { from: string; to: string };\n  /** Total estimated savings */\n  totalSavings: number;\n  /** Breakdown by category */\n  breakdown: {\n    ticketDeflection: number;\n    escalationPrevention: number;\n    timeToResolution: number;\n    churnPrevention: number;\n  };\n  /** Confidence in these estimates */\n  confidence: 'low' | 'medium' | 'high';\n  /** Methodology used */\n  methodology: string;\n}\n\n// ============================================================================\n// EXPLAINABILITY RESULT (v2 compatibility)\n// ============================================================================\n\n/**\n * @deprecated Use AnalysisResult instead\n */\nexport interface ExplainabilityResult {\n  traceId: string;\n  explainabilityId: string;\n  summary: string;\n  outcome: {\n    verdict: 'success' | 'partial_success' | 'failure';\n    confidence: number;\n    reasoning: string;\n  };\n  businessImpact: {\n    impactScore: number;\n    customerSatisfaction: number;\n    revenueRisk: string;\n  };\n  recommendations: Array<{\n    priority: string;\n    category: string;\n    action: string;\n    expectedImpact: string;\n  }>;\n  ragEvaluation?: {\n    groundedness: number;\n    faithfulness: number;\n    answerRelevance: number;\n  };\n  hallucinationReport?: {\n    detected: boolean;\n    types: string[];\n    severity: string;\n  };\n  processingTimeMs: number;\n}\n\n// ============================================================================\n// API RESPONSE TYPES\n// ============================================================================\n\nexport interface ApiResponse<T> {\n  success: boolean;\n  data?: T;\n  error?: {\n    code: string;\n    message: string;\n    details?: unknown;\n  };\n}\n\nexport interface PaginatedResponse<T> {\n  success: boolean;\n  data: {\n    items: T[];\n    limit: number;\n    offset: number;\n    hasMore: boolean;\n  };\n}\n\nexport interface RunResponse {\n  id: string;\n  companyId: string;\n  agentId: string;\n  ticketId?: string;\n  customerAccountId?: string;\n  sessionId?: string;\n  startedAt: string;\n  endedAt?: string;\n  durationMs?: number;\n  outcome?: RunOutcome;\n  outcomeReason?: string;\n  conversationMessages: ConversationMessage[];\n  modelId?: string;\n  promptVersion?: string;\n  customerContextSnapshot?: CustomerContextSnapshot;\n  zendeskMarker?: string;\n  createdAt: string;\n  updatedAt: string;\n}\n"]}
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/core/types.ts"],"names":[],"mappings":";AAAA;;;;GAIG","sourcesContent":["/**\n * ThinkHive SDK v3.0 - Core Types\n *\n * Run-centric types with facts vs inferences and customer context snapshots\n */\n\n// ============================================================================\n// INIT OPTIONS\n// ============================================================================\n\nexport interface InitOptions {\n  /** ThinkHive API key (starts with th_) */\n  apiKey?: string;\n  /** Agent ID */\n  agentId?: string;\n  /** ThinkHive API endpoint */\n  endpoint?: string;\n  /** Service name for traces */\n  serviceName?: string;\n  /** Enable auto-instrumentation */\n  autoInstrument?: boolean;\n  /** Frameworks to auto-instrument */\n  frameworks?: Framework[];\n  /** Enable debug logging */\n  debug?: boolean;\n  /** API version (default: 'v3') */\n  apiVersion?: 'v1' | 'v3';\n}\n\nexport type Framework =\n  | 'langchain'\n  | 'langgraph'\n  | 'openai'\n  | 'openai-assistants'\n  | 'anthropic'\n  | 'llamaindex'\n  | 'n8n'\n  | 'voiceflow';\n\n// ============================================================================\n// RUN OPTIONS (v3)\n// ============================================================================\n\n/**\n * Options for creating a run (v3 atomic unit)\n */\nexport interface RunOptions {\n  /** Agent ID */\n  agentId: string;\n  /** When the run started */\n  startedAt?: string | Date;\n  /** When the run ended */\n  endedAt?: string | Date;\n  /** Run outcome */\n  outcome?: RunOutcome;\n  /** Reason for the outcome */\n  outcomeReason?: string;\n  /** Conversation messages */\n  conversationMessages: ConversationMessage[];\n  /** Model ID used */\n  modelId?: string;\n  /** Prompt version */\n  promptVersion?: string;\n  /** Session ID for conversation tracking */\n  sessionId?: string;\n  /** Customer context snapshot (ARR, health AS OF this run) */\n  customerContext?: CustomerContextSnapshot;\n  /** Ticket linking options */\n  ticketLinking?: TicketLinkingOptions;\n  /** Custom metadata */\n  metadata?: Record<string, unknown>;\n  /**\n   * Request evaluation when run is ingested.\n   * Overrides agent's auto_evaluate setting for this run.\n   *\n   * - true: Force evaluation on this run\n   * - false: Skip evaluation even if agent has auto_evaluate enabled\n   * - undefined: Use agent's auto_evaluate setting (default)\n   */\n  runEvaluation?: boolean;\n}\n\nexport type RunOutcome =\n  | 'resolved'\n  | 'escalated'\n  | 'abandoned'\n  | 'failed'\n  | 'transferred'\n  | 'pending';\n\nexport interface ConversationMessage {\n  role: 'user' | 'assistant' | 'system' | 'tool';\n  content: string;\n  timestamp?: string;\n  metadata?: Record<string, unknown>;\n}\n\n// ============================================================================\n// CUSTOMER CONTEXT SNAPSHOT (v3)\n// ============================================================================\n\n/**\n * Customer context captured AS OF the run time (not current values)\n */\nexport interface CustomerContextSnapshot {\n  /** Customer account ID */\n  customerId?: string;\n  /** ARR at the time of the run (not current ARR) */\n  arr?: number;\n  /** Health score at the time of the run (0-100) */\n  healthScore?: number;\n  /** Customer segment (e.g., 'enterprise', 'mid-market', 'smb') */\n  segment?: string;\n  /** When this snapshot was captured */\n  capturedAt?: string;\n  /** Custom fields (will be validated against schema) */\n  customFields?: Record<string, unknown>;\n}\n\n// ============================================================================\n// TICKET LINKING (v3)\n// ============================================================================\n\n/**\n * Options for deterministic ticket linking\n */\nexport interface TicketLinkingOptions {\n  /** Link method to use */\n  method: LinkMethod;\n  /** Ticket ID (for sdk_explicit method) */\n  ticketId?: string;\n  /** External ticket ID (e.g., Zendesk ticket ID) */\n  externalTicketId?: string;\n  /** Platform for external ticket */\n  platform?: 'zendesk' | 'intercom' | 'salesforce' | 'freshdesk';\n  /** Custom field name for linking */\n  customFieldName?: string;\n  /** Custom field value */\n  customFieldValue?: string;\n}\n\nexport type LinkMethod =\n  | 'sdk_explicit'      // Direct SDK call with ticket ID (confidence: 1.0)\n  | 'zendesk_marker'    // Embedded THID marker in response (confidence: 1.0)\n  | 'custom_field'      // Zendesk custom field (confidence: 1.0)\n  | 'middleware_stamp'  // Middleware-injected trace ID (confidence: 0.98)\n  | 'session_match'     // Session ID correlation (confidence: 0.95)\n  | 'email_time_window' // Email + 15min window (confidence: 0.6)\n  | 'manual';           // Human-assigned (confidence: 1.0)\n\n// ============================================================================\n// TRACE OPTIONS (v2 compatibility)\n// ============================================================================\n\n/**\n * Classification flags for categorizing trace issues\n * Separate from binary outcome (success/failure)\n */\nexport type TraceCustomFlag =\n  | 'hallucination'\n  | 'policy_violation'\n  | 'tone_issue'\n  | 'retrieval_miss'\n  | 'error';\n\n/**\n * Trace options (v2 format, converted to runs internally)\n * @deprecated Use RunOptions instead\n */\nexport interface TraceOptions {\n  /** User's message/query */\n  userMessage: string;\n  /** Agent's response */\n  agentResponse: string;\n  /** User's detected intent */\n  userIntent?: string;\n  /** Binary outcome: success or failure */\n  outcome?: 'success' | 'failure';\n  /** Classification flags for categorizing the issue type (hallucination, policy_violation, etc.) */\n  customFlags?: TraceCustomFlag[];\n  /** Duration in milliseconds */\n  duration?: number;\n  /** Session ID for conversation tracking */\n  sessionId?: string;\n  /** Conversation history */\n  conversationHistory?: Array<{ role: string; content: string }>;\n  /** Span data for detailed analysis */\n  spans?: SpanData[];\n  /** Business context for ROI calculation */\n  businessContext?: BusinessContext;\n  /** Custom metadata */\n  metadata?: Record<string, unknown>;\n  /**\n   * Request evaluation when trace is ingested.\n   * Overrides agent's auto_evaluate setting for this trace.\n   *\n   * - true: Force evaluation on this trace\n   * - false: Skip evaluation even if agent has auto_evaluate enabled\n   * - undefined: Use agent's auto_evaluate setting (default)\n   */\n  runEvaluation?: boolean;\n}\n\nexport interface SpanData {\n  id?: string;\n  name: string;\n  type: 'llm' | 'tool' | 'retrieval' | 'embedding' | 'chain' | 'custom';\n  startTime?: Date;\n  endTime?: Date;\n  durationMs?: number;\n  status?: 'ok' | 'error' | 'timeout';\n  error?: string;\n  input?: unknown;\n  output?: unknown;\n  // LLM-specific\n  model?: string;\n  provider?: string;\n  promptTokens?: number;\n  completionTokens?: number;\n  // Tool-specific\n  toolName?: string;\n  toolParameters?: Record<string, unknown>;\n  // Retrieval-specific\n  query?: string;\n  documentCount?: number;\n  topScore?: number;\n  sources?: string[];\n  // Children\n  children?: SpanData[];\n}\n\n/**\n * @deprecated Use CustomerContextSnapshot instead\n */\nexport interface BusinessContext {\n  /** Customer ID */\n  customerId?: string;\n  /** Transaction value in dollars */\n  transactionValue?: number;\n  /** Priority level */\n  priority?: 'low' | 'medium' | 'high' | 'critical';\n  /** Department */\n  department?: string;\n  /** Industry vertical */\n  industry?: string;\n  /** Custom fields */\n  custom?: Record<string, unknown>;\n}\n\n// ============================================================================\n// CLAIM TYPES (v3)\n// ============================================================================\n\n/**\n * A claim about a run (fact vs inference)\n */\nexport interface Claim {\n  id: string;\n  analysisId: string;\n  /** Type of claim */\n  claimType: ClaimType;\n  /** Category of the claim */\n  claimCategory: ClaimCategory;\n  /** The claim text */\n  claimText: string;\n  /** Confidence in this claim (0-1) */\n  confidence: number;\n  /** How well calibrated is this confidence */\n  confidenceCalibration: ConfidenceCalibration;\n  /** Evidence supporting this claim */\n  evidence: EvidenceReference[];\n  /** Is this claim explainable to end users */\n  isExplainable: boolean;\n  /** Probability value for probability claims */\n  probabilityValue?: number;\n  /** Human-assigned verification status */\n  humanVerified?: boolean;\n  /** Human verification verdict */\n  humanVerdict?: 'confirmed' | 'rejected' | 'modified';\n  createdAt?: string;\n}\n\nexport type ClaimType = 'observed' | 'inferred' | 'computed';\nexport type ClaimCategory =\n  | 'outcome'\n  | 'root_cause'\n  | 'customer_impact'\n  | 'churn_risk'\n  | 'revenue_impact'\n  | 'quality'\n  | 'other';\nexport type ConfidenceCalibration = 'calibrated' | 'uncalibrated' | 'needs_more_data';\n\nexport interface EvidenceReference {\n  /** Type of evidence */\n  type: 'span' | 'message' | 'tool_call' | 'external' | 'computed';\n  /** ID of the referenced item */\n  referenceId: string;\n  /** How relevant is this evidence */\n  relevance: string;\n  /** Confidence in this evidence */\n  confidence: number;\n}\n\n// ============================================================================\n// ANALYSIS RESULT (v3)\n// ============================================================================\n\nexport interface AnalysisResult {\n  id: string;\n  runId: string;\n  analysisVersion: string;\n  modelUsed: string;\n  /** Overall outcome verdict */\n  outcomeVerdict: 'success' | 'partial_success' | 'failure';\n  /** Confidence in the verdict (0-1) */\n  outcomeConfidence: number;\n  /** Root cause category if failure */\n  rootCauseCategory?: string;\n  /** Confidence in root cause (0-1) */\n  rootCauseConfidence?: number;\n  /** Is this the current analysis */\n  isCurrent: boolean;\n  /** ID of analysis that superseded this one */\n  supersededBy?: string;\n  /** Claims from this analysis */\n  claims: Claim[];\n  analyzedAt?: string;\n  createdAt?: string;\n}\n\n// ============================================================================\n// CALIBRATION TYPES (v3)\n// ============================================================================\n\nexport interface CalibrationStatus {\n  agentId: string;\n  predictionType: PredictionType;\n  /** Brier score (lower is better, <0.1 is good) */\n  brierScore: number;\n  /** Expected Calibration Error */\n  ece: number;\n  /** Sample count */\n  sampleCount: number;\n  /** Is the model well-calibrated */\n  isCalibrated: boolean;\n  /** Calibration by confidence bucket */\n  buckets: CalibrationBucket[];\n  lastUpdated?: string;\n}\n\nexport type PredictionType =\n  | 'outcome'\n  | 'churn_risk'\n  | 'escalation_risk'\n  | 'resolution_time'\n  | 'customer_satisfaction';\n\nexport interface CalibrationBucket {\n  bucketStart: number;\n  bucketEnd: number;\n  predictedProbability: number;\n  actualFrequency: number;\n  sampleCount: number;\n}\n\n// ============================================================================\n// ROI TYPES (v3)\n// ============================================================================\n\nexport interface RoiConfig {\n  /** Cost per support ticket deflected */\n  ticketDeflectionValue: number;\n  /** Cost per escalation */\n  escalationCost: number;\n  /** Cost per hour of agent time */\n  agentHourlyCost: number;\n  /** Average resolution time (minutes) */\n  avgResolutionMinutes: number;\n  /** CSAT impact multiplier */\n  csatImpactMultiplier: number;\n}\n\nexport interface RoiSummary {\n  period: { from: string; to: string };\n  /** Total estimated savings */\n  totalSavings: number;\n  /** Breakdown by category */\n  breakdown: {\n    ticketDeflection: number;\n    escalationPrevention: number;\n    timeToResolution: number;\n    churnPrevention: number;\n  };\n  /** Confidence in these estimates */\n  confidence: 'low' | 'medium' | 'high';\n  /** Methodology used */\n  methodology: string;\n}\n\n// ============================================================================\n// EXPLAINABILITY RESULT (v2 compatibility)\n// ============================================================================\n\n/**\n * @deprecated Use AnalysisResult instead\n */\nexport interface ExplainabilityResult {\n  traceId: string;\n  explainabilityId: string;\n  summary: string;\n  outcome: {\n    verdict: 'success' | 'partial_success' | 'failure';\n    confidence: number;\n    reasoning: string;\n  };\n  businessImpact: {\n    impactScore: number;\n    customerSatisfaction: number;\n    revenueRisk: string;\n  };\n  recommendations: Array<{\n    priority: string;\n    category: string;\n    action: string;\n    expectedImpact: string;\n  }>;\n  ragEvaluation?: {\n    groundedness: number;\n    faithfulness: number;\n    answerRelevance: number;\n  };\n  hallucinationReport?: {\n    detected: boolean;\n    types: string[];\n    severity: string;\n  };\n  processingTimeMs: number;\n}\n\n// ============================================================================\n// API RESPONSE TYPES\n// ============================================================================\n\nexport interface ApiResponse<T> {\n  success: boolean;\n  data?: T;\n  error?: {\n    code: string;\n    message: string;\n    details?: unknown;\n  };\n}\n\nexport interface PaginatedResponse<T> {\n  success: boolean;\n  data: {\n    items: T[];\n    limit: number;\n    offset: number;\n    hasMore: boolean;\n  };\n}\n\nexport interface RunResponse {\n  id: string;\n  companyId: string;\n  agentId: string;\n  ticketId?: string;\n  customerAccountId?: string;\n  sessionId?: string;\n  startedAt: string;\n  endedAt?: string;\n  durationMs?: number;\n  outcome?: RunOutcome;\n  outcomeReason?: string;\n  conversationMessages: ConversationMessage[];\n  modelId?: string;\n  promptVersion?: string;\n  customerContextSnapshot?: CustomerContextSnapshot;\n  zendeskMarker?: string;\n  createdAt: string;\n  updatedAt: string;\n}\n"]}
|