@thinkhive/sdk 3.1.1 → 4.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.
@@ -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
- * Make an authenticated API request
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;
18
- const version = apiVersion || config.apiVersion;
19
- const url = `${config.endpoint}/api/${version}${path}`;
40
+ const { method = 'GET', body, headers = {}, apiVersion, maxRetries, timeout } = options;
41
+ const version = apiVersion === 'none' ? '' : (apiVersion || '');
42
+ const url = version ? `${config.endpoint}/api/${version}${path}` : `${config.endpoint}/api${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.0.0',
47
+ 'X-SDK-Version': config_1.SDK_VERSION,
23
48
  ...headers,
24
49
  };
25
50
  if (config.apiKey) {
@@ -29,32 +54,112 @@ async function apiRequest(path, options = {}) {
29
54
  requestHeaders['X-Agent-ID'] = config.agentId;
30
55
  }
31
56
  (0, config_1.debugLog)(`${method} ${url}`);
32
- const response = await fetch(url, {
33
- method,
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
- errorData = JSON.parse(errorText);
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
+ // Handle 204 No Content (e.g., successful DELETE)
119
+ if (response.status === 204) {
120
+ return undefined;
121
+ }
122
+ return response.json();
50
123
  }
51
- catch {
52
- errorData = { message: errorText };
124
+ catch (error) {
125
+ // Handle abort (timeout)
126
+ if (error instanceof Error && error.name === 'AbortError') {
127
+ lastError = new ThinkHiveApiError(`Request timed out after ${requestTimeout}ms`, 408);
128
+ if (attempt < retries) {
129
+ const backoff = calculateBackoff(attempt);
130
+ (0, config_1.debugLog)(`Request timed out, retrying after ${backoff}ms (attempt ${attempt + 1}/${retries})`);
131
+ await sleep(backoff);
132
+ continue;
133
+ }
134
+ throw lastError;
135
+ }
136
+ // Handle network errors (TypeError from fetch)
137
+ if (error instanceof TypeError) {
138
+ lastError = new ThinkHiveApiError(`Network error: ${error.message}`, 503);
139
+ if (attempt < retries) {
140
+ const backoff = calculateBackoff(attempt);
141
+ (0, config_1.debugLog)(`Network error, retrying after ${backoff}ms (attempt ${attempt + 1}/${retries})`);
142
+ await sleep(backoff);
143
+ continue;
144
+ }
145
+ throw lastError;
146
+ }
147
+ // Re-throw ThinkHive errors (non-retryable)
148
+ if (error instanceof ThinkHiveError) {
149
+ throw error;
150
+ }
151
+ // Unknown error
152
+ lastError = error instanceof Error ? error : new Error(String(error));
153
+ if (attempt < retries) {
154
+ const backoff = calculateBackoff(attempt);
155
+ await sleep(backoff);
156
+ continue;
157
+ }
158
+ throw new ThinkHiveApiError(`Request failed: ${lastError.message}`, 500);
53
159
  }
54
- const error = new ThinkHiveApiError(errorData.message || `HTTP ${response.status}`, response.status, errorData.code);
55
- throw error;
56
160
  }
57
- return response.json();
161
+ // Should not reach here, but just in case
162
+ throw lastError || new ThinkHiveApiError('Request failed after all retries', 500);
58
163
  }
59
164
  /**
60
165
  * Make an API request and extract data from response wrapper
@@ -69,21 +174,111 @@ async function apiRequestWithData(path, options = {}) {
69
174
  // ============================================================================
70
175
  // ERRORS
71
176
  // ============================================================================
72
- class ThinkHiveApiError extends Error {
73
- constructor(message, statusCode, code) {
177
+ /**
178
+ * Base error class for ThinkHive SDK errors
179
+ */
180
+ class ThinkHiveError extends Error {
181
+ constructor(message, code, statusCode, details) {
74
182
  super(message);
75
- this.statusCode = statusCode;
76
183
  this.code = code;
184
+ this.statusCode = statusCode;
185
+ this.details = details;
186
+ this.name = 'ThinkHiveError';
187
+ }
188
+ }
189
+ exports.ThinkHiveError = ThinkHiveError;
190
+ /**
191
+ * API request error - returned from ThinkHive server
192
+ */
193
+ class ThinkHiveApiError extends ThinkHiveError {
194
+ constructor(message, statusCode, code) {
195
+ super(message, code || 'API_ERROR', statusCode);
77
196
  this.name = 'ThinkHiveApiError';
78
197
  }
79
198
  }
80
199
  exports.ThinkHiveApiError = ThinkHiveApiError;
81
- class ThinkHiveValidationError extends Error {
200
+ /**
201
+ * Validation error - invalid input parameters
202
+ */
203
+ class ThinkHiveValidationError extends ThinkHiveError {
82
204
  constructor(message, field) {
83
- super(message);
205
+ super(message, 'VALIDATION_ERROR', 400, field ? { field } : undefined);
84
206
  this.field = field;
85
207
  this.name = 'ThinkHiveValidationError';
86
208
  }
87
209
  }
88
210
  exports.ThinkHiveValidationError = ThinkHiveValidationError;
89
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client.js","sourceRoot":"","sources":["../../src/core/client.ts"],"names":[],"mappings":";AAAA;;;;GAIG;;;AAoBH,gCA2DC;AAKD,gDAaC;AA/FD,qCAA+C;AAe/C;;GAEG;AACI,KAAK,UAAU,UAAU,CAC9B,IAAY,EACZ,UAA0B,EAAE;IAE5B,MAAM,MAAM,GAAG,IAAA,kBAAS,GAAE,CAAC;IAC3B,MAAM,EAAE,MAAM,GAAG,KAAK,EAAE,IAAI,EAAE,OAAO,GAAG,EAAE,EAAE,UAAU,EAAE,GAAG,OAAO,CAAC;IAEnE,MAAM,OAAO,GAAG,UAAU,IAAI,MAAM,CAAC,UAAU,CAAC;IAChD,MAAM,GAAG,GAAG,GAAG,MAAM,CAAC,QAAQ,QAAQ,OAAO,GAAG,IAAI,EAAE,CAAC;IAEvD,MAAM,cAAc,GAA2B;QAC7C,cAAc,EAAE,kBAAkB;QAClC,eAAe,EAAE,OAAO;QACxB,GAAG,OAAO;KACX,CAAC;IAEF,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;QAClB,cAAc,CAAC,eAAe,CAAC,GAAG,UAAU,MAAM,CAAC,MAAM,EAAE,CAAC;IAC9D,CAAC;SAAM,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;QAC1B,cAAc,CAAC,YAAY,CAAC,GAAG,MAAM,CAAC,OAAO,CAAC;IAChD,CAAC;IAED,IAAA,iBAAQ,EAAC,GAAG,MAAM,IAAI,GAAG,EAAE,CAAC,CAAC;IAE7B,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE;QAChC,MAAM;QACN,OAAO,EAAE,cAAc;QACvB,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS;KAC9C,CAAC,CAAC;IAEH,8BAA8B;IAC9B,IAAI,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE,CAAC;QACxC,MAAM,MAAM,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC9C,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QAC1C,OAAO,CAAC,IAAI,CACV,+DAA+D;YAC7D,CAAC,MAAM,CAAC,CAAC,CAAC,YAAY,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACpC,CAAC,IAAI,CAAC,CAAC,CAAC,eAAe,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CACtC,CAAC;IACJ,CAAC;IAED,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;QACjB,MAAM,SAAS,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACxC,IAAI,SAAS,GAAwC,EAAE,CAAC;QACxD,IAAI,CAAC;YACH,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QACpC,CAAC;QAAC,MAAM,CAAC;YACP,SAAS,GAAG,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;QACrC,CAAC;QAED,MAAM,KAAK,GAAG,IAAI,iBAAiB,CACjC,SAAS,CAAC,OAAO,IAAI,QAAQ,QAAQ,CAAC,MAAM,EAAE,EAC9C,QAAQ,CAAC,MAAM,EACf,SAAS,CAAC,IAAI,CACf,CAAC;QACF,MAAM,KAAK,CAAC;IACd,CAAC;IAED,OAAO,QAAQ,CAAC,IAAI,EAAgB,CAAC;AACvC,CAAC;AAED;;GAEG;AACI,KAAK,UAAU,kBAAkB,CACtC,IAAY,EACZ,UAA0B,EAAE;IAE5B,MAAM,QAAQ,GAAG,MAAM,UAAU,CAAiB,IAAI,EAAE,OAAO,CAAC,CAAC;IACjE,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC;QACtB,MAAM,IAAI,iBAAiB,CACzB,QAAQ,CAAC,KAAK,EAAE,OAAO,IAAI,eAAe,EAC1C,GAAG,EACH,QAAQ,CAAC,KAAK,EAAE,IAAI,CACrB,CAAC;IACJ,CAAC;IACD,OAAO,QAAQ,CAAC,IAAS,CAAC;AAC5B,CAAC;AAED,+EAA+E;AAC/E,SAAS;AACT,+EAA+E;AAE/E,MAAa,iBAAkB,SAAQ,KAAK;IAC1C,YACE,OAAe,EACC,UAAkB,EAClB,IAAa;QAE7B,KAAK,CAAC,OAAO,CAAC,CAAC;QAHC,eAAU,GAAV,UAAU,CAAQ;QAClB,SAAI,GAAJ,IAAI,CAAS;QAG7B,IAAI,CAAC,IAAI,GAAG,mBAAmB,CAAC;IAClC,CAAC;CACF;AATD,8CASC;AAED,MAAa,wBAAyB,SAAQ,KAAK;IACjD,YACE,OAAe,EACC,KAAc;QAE9B,KAAK,CAAC,OAAO,CAAC,CAAC;QAFC,UAAK,GAAL,KAAK,CAAS;QAG9B,IAAI,CAAC,IAAI,GAAG,0BAA0B,CAAC;IACzC,CAAC;CACF;AARD,4DAQC","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// HTTP CLIENT\n// ============================================================================\n\nexport interface RequestOptions {\n  method?: 'GET' | 'POST' | 'PUT' | 'DELETE';\n  body?: unknown;\n  headers?: Record<string, string>;\n  /** Override API version for this request */\n  apiVersion?: 'v1' | 'v3';\n}\n\n/**\n * Make an authenticated API request\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 } = options;\n\n  const version = apiVersion || config.apiVersion;\n  const url = `${config.endpoint}/api/${version}${path}`;\n\n  const requestHeaders: Record<string, string> = {\n    'Content-Type': 'application/json',\n    'X-SDK-Version': '3.0.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  const response = await fetch(url, {\n    method,\n    headers: requestHeaders,\n    body: body ? JSON.stringify(body) : undefined,\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  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    const error = new ThinkHiveApiError(\n      errorData.message || `HTTP ${response.status}`,\n      response.status,\n      errorData.code\n    );\n    throw error;\n  }\n\n  return response.json() as Promise<T>;\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\nexport class ThinkHiveApiError extends Error {\n  constructor(\n    message: string,\n    public readonly statusCode: number,\n    public readonly code?: string\n  ) {\n    super(message);\n    this.name = 'ThinkHiveApiError';\n  }\n}\n\nexport class ThinkHiveValidationError extends Error {\n  constructor(\n    message: string,\n    public readonly field?: string\n  ) {\n    super(message);\n    this.name = 'ThinkHiveValidationError';\n  }\n}\n"]}
211
+ /**
212
+ * Permission denied error - API key lacks required permissions
213
+ * @example
214
+ * ```typescript
215
+ * try {
216
+ * await thinkHive.traces.create(...);
217
+ * } catch (error) {
218
+ * if (error instanceof PermissionDeniedError) {
219
+ * console.log('API key needs write permission');
220
+ * }
221
+ * }
222
+ * ```
223
+ */
224
+ class PermissionDeniedError extends ThinkHiveError {
225
+ constructor(message, details) {
226
+ super(message, 'PERMISSION_DENIED', 403, details);
227
+ this.name = 'PermissionDeniedError';
228
+ }
229
+ }
230
+ exports.PermissionDeniedError = PermissionDeniedError;
231
+ /**
232
+ * Agent scope violation error - API key not authorized for this agent
233
+ * @example
234
+ * ```typescript
235
+ * // Create key scoped to specific agents
236
+ * const key = await thinkHive.apiKeys.create({
237
+ * name: 'Agent A Key',
238
+ * allowedAgentIds: ['agent-a-id']
239
+ * });
240
+ *
241
+ * // Attempting to use for Agent B will throw AgentScopeError
242
+ * ```
243
+ */
244
+ class AgentScopeError extends ThinkHiveError {
245
+ constructor(agentId, allowedAgents) {
246
+ super(`API key not authorized for agent ${agentId}`, 'AGENT_SCOPE_VIOLATION', 403, { agentId, allowedAgents });
247
+ this.name = 'AgentScopeError';
248
+ }
249
+ }
250
+ exports.AgentScopeError = AgentScopeError;
251
+ /**
252
+ * Rate limit exceeded error
253
+ * @example
254
+ * ```typescript
255
+ * try {
256
+ * await thinkHive.traces.create(...);
257
+ * } catch (error) {
258
+ * if (error instanceof RateLimitError) {
259
+ * // Wait and retry
260
+ * await sleep(error.retryAfter);
261
+ * await thinkHive.traces.create(...);
262
+ * }
263
+ * }
264
+ * ```
265
+ */
266
+ class RateLimitError extends ThinkHiveError {
267
+ constructor(retryAfter) {
268
+ super(`Rate limit exceeded. Retry after ${retryAfter}ms`, 'RATE_LIMIT_EXCEEDED', 429, { retryAfter });
269
+ this.retryAfter = retryAfter;
270
+ this.name = 'RateLimitError';
271
+ }
272
+ }
273
+ exports.RateLimitError = RateLimitError;
274
+ /**
275
+ * IP whitelist violation error
276
+ */
277
+ class IpWhitelistError extends ThinkHiveError {
278
+ constructor(clientIp) {
279
+ super('Request IP address is not authorized for this API key', 'IP_WHITELIST_VIOLATION', 403, { clientIp });
280
+ this.name = 'IpWhitelistError';
281
+ }
282
+ }
283
+ exports.IpWhitelistError = IpWhitelistError;
284
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client.js","sourceRoot":"","sources":["../../src/core/client.ts"],"names":[],"mappings":";AAAA;;;;GAIG;;;AAmDH,gCAqJC;AAKD,gDAaC;AAxND,qCAA4D;AAG5D,+EAA+E;AAC/E,sBAAsB;AACtB,+EAA+E;AAE/E,MAAM,WAAW,GAAG,CAAC,CAAC;AACtB,MAAM,eAAe,GAAG,GAAG,CAAC,CAAC,KAAK;AAClC,MAAM,WAAW,GAAG,IAAI,CAAC,CAAC,KAAK;AAC/B,MAAM,eAAe,GAAG,IAAI,GAAG,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC,CAAC;AAEhE;;GAEG;AACH,SAAS,gBAAgB,CAAC,OAAe,EAAE,UAAmB;IAC5D,IAAI,UAAU,KAAK,SAAS,EAAE,CAAC;QAC7B,OAAO,IAAI,CAAC,GAAG,CAAC,UAAU,GAAG,IAAI,EAAE,WAAW,CAAC,CAAC,CAAC,wBAAwB;IAC3E,CAAC;IACD,MAAM,OAAO,GAAG,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;IACvD,OAAO,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,WAAW,CAAC,CAAC;AACxC,CAAC;AAED;;GAEG;AACH,SAAS,KAAK,CAAC,EAAU;IACvB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC;AAC3D,CAAC;AAkBD;;GAEG;AACI,KAAK,UAAU,UAAU,CAC9B,IAAY,EACZ,UAA0B,EAAE;IAE5B,MAAM,MAAM,GAAG,IAAA,kBAAS,GAAE,CAAC;IAC3B,MAAM,EAAE,MAAM,GAAG,KAAK,EAAE,IAAI,EAAE,OAAO,GAAG,EAAE,EAAE,UAAU,EAAE,UAAU,EAAE,OAAO,EAAE,GAAG,OAAO,CAAC;IAExF,MAAM,OAAO,GAAG,UAAU,KAAK,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,UAAU,IAAI,EAAE,CAAC,CAAC;IAChE,MAAM,GAAG,GAAG,OAAO,CAAC,CAAC,CAAC,GAAG,MAAM,CAAC,QAAQ,QAAQ,OAAO,GAAG,IAAI,EAAE,CAAC,CAAC,CAAC,GAAG,MAAM,CAAC,QAAQ,OAAO,IAAI,EAAE,CAAC;IACnG,MAAM,OAAO,GAAG,UAAU,IAAI,WAAW,CAAC;IAC1C,MAAM,cAAc,GAAG,OAAO,IAAI,KAAK,CAAC,CAAC,cAAc;IAEvD,MAAM,cAAc,GAA2B;QAC7C,cAAc,EAAE,kBAAkB;QAClC,eAAe,EAAE,oBAAW;QAC5B,GAAG,OAAO;KACX,CAAC;IAEF,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;QAClB,cAAc,CAAC,eAAe,CAAC,GAAG,UAAU,MAAM,CAAC,MAAM,EAAE,CAAC;IAC9D,CAAC;SAAM,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;QAC1B,cAAc,CAAC,YAAY,CAAC,GAAG,MAAM,CAAC,OAAO,CAAC;IAChD,CAAC;IAED,IAAA,iBAAQ,EAAC,GAAG,MAAM,IAAI,GAAG,EAAE,CAAC,CAAC;IAE7B,IAAI,SAAS,GAAiB,IAAI,CAAC;IAEnC,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,IAAI,OAAO,EAAE,OAAO,EAAE,EAAE,CAAC;QACpD,IAAI,CAAC;YACH,sCAAsC;YACtC,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;YACzC,MAAM,SAAS,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,cAAc,CAAC,CAAC;YAEvE,IAAI,QAAkB,CAAC;YACvB,IAAI,CAAC;gBACH,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE;oBAC1B,MAAM;oBACN,OAAO,EAAE,cAAc;oBACvB,IAAI,EAAE,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS;oBAC7C,MAAM,EAAE,UAAU,CAAC,MAAM;iBAC1B,CAAC,CAAC;YACL,CAAC;oBAAS,CAAC;gBACT,YAAY,CAAC,SAAS,CAAC,CAAC;YAC1B,CAAC;YAED,8BAA8B;YAC9B,IAAI,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC,EAAE,CAAC;gBACxC,MAAM,MAAM,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;gBAC9C,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;gBAC1C,OAAO,CAAC,IAAI,CACV,+DAA+D;oBAC7D,CAAC,MAAM,CAAC,CAAC,CAAC,YAAY,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;oBACpC,CAAC,IAAI,CAAC,CAAC,CAAC,eAAe,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CACtC,CAAC;YACJ,CAAC;YAED,iCAAiC;YACjC,IAAI,eAAe,CAAC,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,OAAO,GAAG,OAAO,EAAE,CAAC;gBAC9D,wDAAwD;gBACxD,IAAI,UAA8B,CAAC;gBACnC,IAAI,QAAQ,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;oBAC5B,MAAM,gBAAgB,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;oBAC7D,IAAI,gBAAgB,EAAE,CAAC;wBACrB,UAAU,GAAG,UAAU,CAAC,gBAAgB,CAAC,CAAC;wBAC1C,IAAI,KAAK,CAAC,UAAU,CAAC,EAAE,CAAC;4BACtB,UAAU,GAAG,SAAS,CAAC;wBACzB,CAAC;oBACH,CAAC;gBACH,CAAC;gBAED,MAAM,OAAO,GAAG,gBAAgB,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;gBACtD,IAAA,iBAAQ,EAAC,kBAAkB,OAAO,eAAe,OAAO,GAAG,CAAC,IAAI,OAAO,GAAG,CAAC,CAAC;gBAC5E,MAAM,KAAK,CAAC,OAAO,CAAC,CAAC;gBACrB,SAAS;YACX,CAAC;YAED,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;gBACjB,MAAM,SAAS,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;gBACxC,IAAI,SAAS,GAAwC,EAAE,CAAC;gBACxD,IAAI,CAAC;oBACH,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;gBACpC,CAAC;gBAAC,MAAM,CAAC;oBACP,SAAS,GAAG,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;gBACrC,CAAC;gBAED,iCAAiC;gBACjC,IAAI,QAAQ,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;oBAC5B,MAAM,gBAAgB,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC;oBAC7D,MAAM,UAAU,GAAG,gBAAgB,CAAC,CAAC,CAAC,UAAU,CAAC,gBAAgB,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC;oBAClF,MAAM,IAAI,cAAc,CAAC,UAAU,CAAC,CAAC;gBACvC,CAAC;gBAED,MAAM,IAAI,iBAAiB,CACzB,SAAS,CAAC,OAAO,IAAI,QAAQ,QAAQ,CAAC,MAAM,EAAE,EAC9C,QAAQ,CAAC,MAAM,EACf,SAAS,CAAC,IAAI,CACf,CAAC;YACJ,CAAC;YAED,kDAAkD;YAClD,IAAI,QAAQ,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;gBAC5B,OAAO,SAAyB,CAAC;YACnC,CAAC;YAED,OAAO,QAAQ,CAAC,IAAI,EAAgB,CAAC;QACvC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,yBAAyB;YACzB,IAAI,KAAK,YAAY,KAAK,IAAI,KAAK,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;gBAC1D,SAAS,GAAG,IAAI,iBAAiB,CAAC,2BAA2B,cAAc,IAAI,EAAE,GAAG,CAAC,CAAC;gBACtF,IAAI,OAAO,GAAG,OAAO,EAAE,CAAC;oBACtB,MAAM,OAAO,GAAG,gBAAgB,CAAC,OAAO,CAAC,CAAC;oBAC1C,IAAA,iBAAQ,EAAC,qCAAqC,OAAO,eAAe,OAAO,GAAG,CAAC,IAAI,OAAO,GAAG,CAAC,CAAC;oBAC/F,MAAM,KAAK,CAAC,OAAO,CAAC,CAAC;oBACrB,SAAS;gBACX,CAAC;gBACD,MAAM,SAAS,CAAC;YAClB,CAAC;YAED,+CAA+C;YAC/C,IAAI,KAAK,YAAY,SAAS,EAAE,CAAC;gBAC/B,SAAS,GAAG,IAAI,iBAAiB,CAAC,kBAAkB,KAAK,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC;gBAC1E,IAAI,OAAO,GAAG,OAAO,EAAE,CAAC;oBACtB,MAAM,OAAO,GAAG,gBAAgB,CAAC,OAAO,CAAC,CAAC;oBAC1C,IAAA,iBAAQ,EAAC,iCAAiC,OAAO,eAAe,OAAO,GAAG,CAAC,IAAI,OAAO,GAAG,CAAC,CAAC;oBAC3F,MAAM,KAAK,CAAC,OAAO,CAAC,CAAC;oBACrB,SAAS;gBACX,CAAC;gBACD,MAAM,SAAS,CAAC;YAClB,CAAC;YAED,4CAA4C;YAC5C,IAAI,KAAK,YAAY,cAAc,EAAE,CAAC;gBACpC,MAAM,KAAK,CAAC;YACd,CAAC;YAED,gBAAgB;YAChB,SAAS,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;YACtE,IAAI,OAAO,GAAG,OAAO,EAAE,CAAC;gBACtB,MAAM,OAAO,GAAG,gBAAgB,CAAC,OAAO,CAAC,CAAC;gBAC1C,MAAM,KAAK,CAAC,OAAO,CAAC,CAAC;gBACrB,SAAS;YACX,CAAC;YACD,MAAM,IAAI,iBAAiB,CAAC,mBAAmB,SAAS,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC;QAC3E,CAAC;IACH,CAAC;IAED,0CAA0C;IAC1C,MAAM,SAAS,IAAI,IAAI,iBAAiB,CAAC,kCAAkC,EAAE,GAAG,CAAC,CAAC;AACpF,CAAC;AAED;;GAEG;AACI,KAAK,UAAU,kBAAkB,CACtC,IAAY,EACZ,UAA0B,EAAE;IAE5B,MAAM,QAAQ,GAAG,MAAM,UAAU,CAAiB,IAAI,EAAE,OAAO,CAAC,CAAC;IACjE,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,CAAC;QACtB,MAAM,IAAI,iBAAiB,CACzB,QAAQ,CAAC,KAAK,EAAE,OAAO,IAAI,eAAe,EAC1C,GAAG,EACH,QAAQ,CAAC,KAAK,EAAE,IAAI,CACrB,CAAC;IACJ,CAAC;IACD,OAAO,QAAQ,CAAC,IAAS,CAAC;AAC5B,CAAC;AAED,+EAA+E;AAC/E,SAAS;AACT,+EAA+E;AAE/E;;GAEG;AACH,MAAa,cAAe,SAAQ,KAAK;IACvC,YACE,OAAe,EACC,IAAY,EACZ,UAAmB,EACnB,OAAiC;QAEjD,KAAK,CAAC,OAAO,CAAC,CAAC;QAJC,SAAI,GAAJ,IAAI,CAAQ;QACZ,eAAU,GAAV,UAAU,CAAS;QACnB,YAAO,GAAP,OAAO,CAA0B;QAGjD,IAAI,CAAC,IAAI,GAAG,gBAAgB,CAAC;IAC/B,CAAC;CACF;AAVD,wCAUC;AAED;;GAEG;AACH,MAAa,iBAAkB,SAAQ,cAAc;IACnD,YACE,OAAe,EACf,UAAkB,EAClB,IAAa;QAEb,KAAK,CAAC,OAAO,EAAE,IAAI,IAAI,WAAW,EAAE,UAAU,CAAC,CAAC;QAChD,IAAI,CAAC,IAAI,GAAG,mBAAmB,CAAC;IAClC,CAAC;CACF;AATD,8CASC;AAED;;GAEG;AACH,MAAa,wBAAyB,SAAQ,cAAc;IAC1D,YACE,OAAe,EACC,KAAc;QAE9B,KAAK,CAAC,OAAO,EAAE,kBAAkB,EAAE,GAAG,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;QAFvD,UAAK,GAAL,KAAK,CAAS;QAG9B,IAAI,CAAC,IAAI,GAAG,0BAA0B,CAAC;IACzC,CAAC;CACF;AARD,4DAQC;AAED;;;;;;;;;;;;GAYG;AACH,MAAa,qBAAsB,SAAQ,cAAc;IACvD,YAAY,OAAe,EAAE,OAAiC;QAC5D,KAAK,CAAC,OAAO,EAAE,mBAAmB,EAAE,GAAG,EAAE,OAAO,CAAC,CAAC;QAClD,IAAI,CAAC,IAAI,GAAG,uBAAuB,CAAC;IACtC,CAAC;CACF;AALD,sDAKC;AAED;;;;;;;;;;;;GAYG;AACH,MAAa,eAAgB,SAAQ,cAAc;IACjD,YACE,OAAe,EACf,aAAuB;QAEvB,KAAK,CACH,oCAAoC,OAAO,EAAE,EAC7C,uBAAuB,EACvB,GAAG,EACH,EAAE,OAAO,EAAE,aAAa,EAAE,CAC3B,CAAC;QACF,IAAI,CAAC,IAAI,GAAG,iBAAiB,CAAC;IAChC,CAAC;CACF;AAbD,0CAaC;AAED;;;;;;;;;;;;;;GAcG;AACH,MAAa,cAAe,SAAQ,cAAc;IAChD,YACkB,UAAkB;QAElC,KAAK,CACH,oCAAoC,UAAU,IAAI,EAClD,qBAAqB,EACrB,GAAG,EACH,EAAE,UAAU,EAAE,CACf,CAAC;QAPc,eAAU,GAAV,UAAU,CAAQ;QAQlC,IAAI,CAAC,IAAI,GAAG,gBAAgB,CAAC;IAC/B,CAAC;CACF;AAZD,wCAYC;AAED;;GAEG;AACH,MAAa,gBAAiB,SAAQ,cAAc;IAClD,YAAY,QAAgB;QAC1B,KAAK,CACH,uDAAuD,EACvD,wBAAwB,EACxB,GAAG,EACH,EAAE,QAAQ,EAAE,CACb,CAAC;QACF,IAAI,CAAC,IAAI,GAAG,kBAAkB,CAAC;IACjC,CAAC;CACF;AAVD,4CAUC","sourcesContent":["/**\n * ThinkHive SDK v3.0 - HTTP Client\n *\n * Centralized HTTP client with authentication and error handling\n */\n\nimport { getConfig, debugLog, SDK_VERSION } 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. Use 'none' for unversioned routes (e.g., /api/agents). */\n  apiVersion?: 'v1' | 'v2' | 'v3' | 'none';\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 === 'none' ? '' : (apiVersion || '');\n  const url = version ? `${config.endpoint}/api/${version}${path}` : `${config.endpoint}/api${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': SDK_VERSION,\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      // Handle 204 No Content (e.g., successful DELETE)\n      if (response.status === 204) {\n        return undefined as unknown as T;\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"]}
@@ -2,8 +2,8 @@
2
2
  * ThinkHive SDK v3.0 - Configuration
3
3
  */
4
4
  import type { InitOptions, Framework } from './types';
5
- export declare const SDK_VERSION = "3.0.0";
6
- export declare const DEFAULT_ENDPOINT = "https://thinkhivemind-h25z7pvd3q-uc.a.run.app";
5
+ export declare const SDK_VERSION = "4.0.0";
6
+ export declare const DEFAULT_ENDPOINT = "https://app.thinkhive.ai";
7
7
  export declare const DEFAULT_SERVICE_NAME = "my-ai-agent";
8
8
  export interface ResolvedConfig {
9
9
  apiKey: string;
@@ -13,7 +13,6 @@ export interface ResolvedConfig {
13
13
  autoInstrument: boolean;
14
14
  frameworks: Framework[];
15
15
  debug: boolean;
16
- apiVersion: 'v1' | 'v3';
17
16
  }
18
17
  /**
19
18
  * Initialize configuration from options
@@ -12,8 +12,8 @@ exports.debugLog = debugLog;
12
12
  // ============================================================================
13
13
  // CONSTANTS
14
14
  // ============================================================================
15
- exports.SDK_VERSION = '3.0.0';
16
- exports.DEFAULT_ENDPOINT = 'https://thinkhivemind-h25z7pvd3q-uc.a.run.app';
15
+ exports.SDK_VERSION = '4.0.0';
16
+ exports.DEFAULT_ENDPOINT = 'https://app.thinkhive.ai';
17
17
  exports.DEFAULT_SERVICE_NAME = 'my-ai-agent';
18
18
  let config = null;
19
19
  let initialized = false;
@@ -37,7 +37,6 @@ function initConfig(options = {}) {
37
37
  autoInstrument: options.autoInstrument ?? false,
38
38
  frameworks: options.frameworks || ['langchain', 'openai'],
39
39
  debug: options.debug ?? false,
40
- apiVersion: options.apiVersion ?? 'v3',
41
40
  };
42
41
  initialized = true;
43
42
  return config;
@@ -73,4 +72,4 @@ function debugLog(...args) {
73
72
  console.log('[ThinkHive]', ...args);
74
73
  }
75
74
  }
76
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvcmUvY29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7R0FFRzs7O0FBcUNILGdDQXVCQztBQU1ELDhCQUtDO0FBS0Qsc0NBRUM7QUFLRCxrQ0FHQztBQUtELDRCQUlDO0FBM0ZELCtFQUErRTtBQUMvRSxZQUFZO0FBQ1osK0VBQStFO0FBRWxFLFFBQUEsV0FBVyxHQUFHLE9BQU8sQ0FBQztBQUN0QixRQUFBLGdCQUFnQixHQUFHLCtDQUErQyxDQUFDO0FBQ25FLFFBQUEsb0JBQW9CLEdBQUcsYUFBYSxDQUFDO0FBaUJsRCxJQUFJLE1BQU0sR0FBMEIsSUFBSSxDQUFDO0FBQ3pDLElBQUksV0FBVyxHQUFHLEtBQUssQ0FBQztBQUV4QiwrRUFBK0U7QUFDL0UsbUJBQW1CO0FBQ25CLCtFQUErRTtBQUUvRTs7R0FFRztBQUNILFNBQWdCLFVBQVUsQ0FBQyxVQUF1QixFQUFFO0lBQ2xELE1BQU0sTUFBTSxHQUFHLE9BQU8sQ0FBQyxNQUFNLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxpQkFBaUIsSUFBSSxFQUFFLENBQUM7SUFDckUsTUFBTSxPQUFPLEdBQUcsT0FBTyxDQUFDLE9BQU8sSUFBSSxPQUFPLENBQUMsR0FBRyxDQUFDLGtCQUFrQixJQUFJLEVBQUUsQ0FBQztJQUV4RSxJQUFJLENBQUMsTUFBTSxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDeEIsTUFBTSxJQUFJLEtBQUssQ0FDYiw4RUFBOEUsQ0FDL0UsQ0FBQztJQUNKLENBQUM7SUFFRCxNQUFNLEdBQUc7UUFDUCxNQUFNO1FBQ04sT0FBTztRQUNQLFFBQVEsRUFBRSxPQUFPLENBQUMsUUFBUSxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsa0JBQWtCLElBQUksd0JBQWdCO1FBQ2hGLFdBQVcsRUFBRSxPQUFPLENBQUMsV0FBVyxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsc0JBQXNCLElBQUksNEJBQW9CO1FBQzlGLGNBQWMsRUFBRSxPQUFPLENBQUMsY0FBYyxJQUFJLEtBQUs7UUFDL0MsVUFBVSxFQUFFLE9BQU8sQ0FBQyxVQUFVLElBQUksQ0FBQyxXQUFXLEVBQUUsUUFBUSxDQUFDO1FBQ3pELEtBQUssRUFBRSxPQUFPLENBQUMsS0FBSyxJQUFJLEtBQUs7UUFDN0IsVUFBVSxFQUFFLE9BQU8sQ0FBQyxVQUFVLElBQUksSUFBSTtLQUN2QyxDQUFDO0lBRUYsV0FBVyxHQUFHLElBQUksQ0FBQztJQUNuQixPQUFPLE1BQU0sQ0FBQztBQUNoQixDQUFDO0FBRUQ7OztHQUdHO0FBQ0gsU0FBZ0IsU0FBUztJQUN2QixJQUFJLENBQUMsTUFBTSxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7UUFDNUIsTUFBTSxJQUFJLEtBQUssQ0FBQyxtREFBbUQsQ0FBQyxDQUFDO0lBQ3ZFLENBQUM7SUFDRCxPQUFPLE1BQU0sQ0FBQztBQUNoQixDQUFDO0FBRUQ7O0dBRUc7QUFDSCxTQUFnQixhQUFhO0lBQzNCLE9BQU8sV0FBVyxDQUFDO0FBQ3JCLENBQUM7QUFFRDs7R0FFRztBQUNILFNBQWdCLFdBQVc7SUFDekIsTUFBTSxHQUFHLElBQUksQ0FBQztJQUNkLFdBQVcsR0FBRyxLQUFLLENBQUM7QUFDdEIsQ0FBQztBQUVEOztHQUVHO0FBQ0gsU0FBZ0IsUUFBUSxDQUFDLEdBQUcsSUFBZTtJQUN6QyxJQUFJLE1BQU0sRUFBRSxLQUFLLEVBQUUsQ0FBQztRQUNsQixPQUFPLENBQUMsR0FBRyxDQUFDLGFBQWEsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDO0lBQ3RDLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBUaGlua0hpdmUgU0RLIHYzLjAgLSBDb25maWd1cmF0aW9uXG4gKi9cblxuaW1wb3J0IHR5cGUgeyBJbml0T3B0aW9ucywgRnJhbWV3b3JrIH0gZnJvbSAnLi90eXBlcyc7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIENPTlNUQU5UU1xuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG5leHBvcnQgY29uc3QgU0RLX1ZFUlNJT04gPSAnMy4wLjAnO1xuZXhwb3J0IGNvbnN0IERFRkFVTFRfRU5EUE9JTlQgPSAnaHR0cHM6Ly90aGlua2hpdmVtaW5kLWgyNXo3cHZkM3EtdWMuYS5ydW4uYXBwJztcbmV4cG9ydCBjb25zdCBERUZBVUxUX1NFUlZJQ0VfTkFNRSA9ICdteS1haS1hZ2VudCc7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIEdMT0JBTCBDT05GSUcgU1RBVEVcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0IGludGVyZmFjZSBSZXNvbHZlZENvbmZpZyB7XG4gIGFwaUtleTogc3RyaW5nO1xuICBhZ2VudElkOiBzdHJpbmc7XG4gIGVuZHBvaW50OiBzdHJpbmc7XG4gIHNlcnZpY2VOYW1lOiBzdHJpbmc7XG4gIGF1dG9JbnN0cnVtZW50OiBib29sZWFuO1xuICBmcmFtZXdvcmtzOiBGcmFtZXdvcmtbXTtcbiAgZGVidWc6IGJvb2xlYW47XG4gIGFwaVZlcnNpb246ICd2MScgfCAndjMnO1xufVxuXG5sZXQgY29uZmlnOiBSZXNvbHZlZENvbmZpZyB8IG51bGwgPSBudWxsO1xubGV0IGluaXRpYWxpemVkID0gZmFsc2U7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIENPTkZJRyBGVU5DVElPTlNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBJbml0aWFsaXplIGNvbmZpZ3VyYXRpb24gZnJvbSBvcHRpb25zXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpbml0Q29uZmlnKG9wdGlvbnM6IEluaXRPcHRpb25zID0ge30pOiBSZXNvbHZlZENvbmZpZyB7XG4gIGNvbnN0IGFwaUtleSA9IG9wdGlvbnMuYXBpS2V5IHx8IHByb2Nlc3MuZW52LlRISU5LSElWRV9BUElfS0VZIHx8ICcnO1xuICBjb25zdCBhZ2VudElkID0gb3B0aW9ucy5hZ2VudElkIHx8IHByb2Nlc3MuZW52LlRISU5LSElWRV9BR0VOVF9JRCB8fCAnJztcblxuICBpZiAoIWFwaUtleSAmJiAhYWdlbnRJZCkge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICdFaXRoZXIgYXBpS2V5IG9yIGFnZW50SWQgbXVzdCBiZSBwcm92aWRlZCAob3Igc2V0IFRISU5LSElWRV9BUElfS0VZIGVudiB2YXIpJ1xuICAgICk7XG4gIH1cblxuICBjb25maWcgPSB7XG4gICAgYXBpS2V5LFxuICAgIGFnZW50SWQsXG4gICAgZW5kcG9pbnQ6IG9wdGlvbnMuZW5kcG9pbnQgfHwgcHJvY2Vzcy5lbnYuVEhJTktISVZFX0VORFBPSU5UIHx8IERFRkFVTFRfRU5EUE9JTlQsXG4gICAgc2VydmljZU5hbWU6IG9wdGlvbnMuc2VydmljZU5hbWUgfHwgcHJvY2Vzcy5lbnYuVEhJTktISVZFX1NFUlZJQ0VfTkFNRSB8fCBERUZBVUxUX1NFUlZJQ0VfTkFNRSxcbiAgICBhdXRvSW5zdHJ1bWVudDogb3B0aW9ucy5hdXRvSW5zdHJ1bWVudCA/PyBmYWxzZSxcbiAgICBmcmFtZXdvcmtzOiBvcHRpb25zLmZyYW1ld29ya3MgfHwgWydsYW5nY2hhaW4nLCAnb3BlbmFpJ10sXG4gICAgZGVidWc6IG9wdGlvbnMuZGVidWcgPz8gZmFsc2UsXG4gICAgYXBpVmVyc2lvbjogb3B0aW9ucy5hcGlWZXJzaW9uID8/ICd2MycsXG4gIH07XG5cbiAgaW5pdGlhbGl6ZWQgPSB0cnVlO1xuICByZXR1cm4gY29uZmlnO1xufVxuXG4vKipcbiAqIEdldCBjdXJyZW50IGNvbmZpZ3VyYXRpb25cbiAqIEB0aHJvd3MgaWYgbm90IGluaXRpYWxpemVkXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRDb25maWcoKTogUmVzb2x2ZWRDb25maWcge1xuICBpZiAoIWNvbmZpZyB8fCAhaW5pdGlhbGl6ZWQpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ1RoaW5rSGl2ZSBTREsgbm90IGluaXRpYWxpemVkLiBDYWxsIGluaXQoKSBmaXJzdC4nKTtcbiAgfVxuICByZXR1cm4gY29uZmlnO1xufVxuXG4vKipcbiAqIENoZWNrIGlmIFNESyBpcyBpbml0aWFsaXplZFxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNJbml0aWFsaXplZCgpOiBib29sZWFuIHtcbiAgcmV0dXJuIGluaXRpYWxpemVkO1xufVxuXG4vKipcbiAqIFJlc2V0IGNvbmZpZ3VyYXRpb24gKGZvciB0ZXN0aW5nKVxuICovXG5leHBvcnQgZnVuY3Rpb24gcmVzZXRDb25maWcoKTogdm9pZCB7XG4gIGNvbmZpZyA9IG51bGw7XG4gIGluaXRpYWxpemVkID0gZmFsc2U7XG59XG5cbi8qKlxuICogRGVidWcgbG9nIGhlbHBlclxuICovXG5leHBvcnQgZnVuY3Rpb24gZGVidWdMb2coLi4uYXJnczogdW5rbm93bltdKTogdm9pZCB7XG4gIGlmIChjb25maWc/LmRlYnVnKSB7XG4gICAgY29uc29sZS5sb2coJ1tUaGlua0hpdmVdJywgLi4uYXJncyk7XG4gIH1cbn1cbiJdfQ==
75
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvcmUvY29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7R0FFRzs7O0FBb0NILGdDQXNCQztBQU1ELDhCQUtDO0FBS0Qsc0NBRUM7QUFLRCxrQ0FHQztBQUtELDRCQUlDO0FBekZELCtFQUErRTtBQUMvRSxZQUFZO0FBQ1osK0VBQStFO0FBRWxFLFFBQUEsV0FBVyxHQUFHLE9BQU8sQ0FBQztBQUN0QixRQUFBLGdCQUFnQixHQUFHLDBCQUEwQixDQUFDO0FBQzlDLFFBQUEsb0JBQW9CLEdBQUcsYUFBYSxDQUFDO0FBZ0JsRCxJQUFJLE1BQU0sR0FBMEIsSUFBSSxDQUFDO0FBQ3pDLElBQUksV0FBVyxHQUFHLEtBQUssQ0FBQztBQUV4QiwrRUFBK0U7QUFDL0UsbUJBQW1CO0FBQ25CLCtFQUErRTtBQUUvRTs7R0FFRztBQUNILFNBQWdCLFVBQVUsQ0FBQyxVQUF1QixFQUFFO0lBQ2xELE1BQU0sTUFBTSxHQUFHLE9BQU8sQ0FBQyxNQUFNLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxpQkFBaUIsSUFBSSxFQUFFLENBQUM7SUFDckUsTUFBTSxPQUFPLEdBQUcsT0FBTyxDQUFDLE9BQU8sSUFBSSxPQUFPLENBQUMsR0FBRyxDQUFDLGtCQUFrQixJQUFJLEVBQUUsQ0FBQztJQUV4RSxJQUFJLENBQUMsTUFBTSxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDeEIsTUFBTSxJQUFJLEtBQUssQ0FDYiw4RUFBOEUsQ0FDL0UsQ0FBQztJQUNKLENBQUM7SUFFRCxNQUFNLEdBQUc7UUFDUCxNQUFNO1FBQ04sT0FBTztRQUNQLFFBQVEsRUFBRSxPQUFPLENBQUMsUUFBUSxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsa0JBQWtCLElBQUksd0JBQWdCO1FBQ2hGLFdBQVcsRUFBRSxPQUFPLENBQUMsV0FBVyxJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsc0JBQXNCLElBQUksNEJBQW9CO1FBQzlGLGNBQWMsRUFBRSxPQUFPLENBQUMsY0FBYyxJQUFJLEtBQUs7UUFDL0MsVUFBVSxFQUFFLE9BQU8sQ0FBQyxVQUFVLElBQUksQ0FBQyxXQUFXLEVBQUUsUUFBUSxDQUFDO1FBQ3pELEtBQUssRUFBRSxPQUFPLENBQUMsS0FBSyxJQUFJLEtBQUs7S0FDOUIsQ0FBQztJQUVGLFdBQVcsR0FBRyxJQUFJLENBQUM7SUFDbkIsT0FBTyxNQUFNLENBQUM7QUFDaEIsQ0FBQztBQUVEOzs7R0FHRztBQUNILFNBQWdCLFNBQVM7SUFDdkIsSUFBSSxDQUFDLE1BQU0sSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDO1FBQzVCLE1BQU0sSUFBSSxLQUFLLENBQUMsbURBQW1ELENBQUMsQ0FBQztJQUN2RSxDQUFDO0lBQ0QsT0FBTyxNQUFNLENBQUM7QUFDaEIsQ0FBQztBQUVEOztHQUVHO0FBQ0gsU0FBZ0IsYUFBYTtJQUMzQixPQUFPLFdBQVcsQ0FBQztBQUNyQixDQUFDO0FBRUQ7O0dBRUc7QUFDSCxTQUFnQixXQUFXO0lBQ3pCLE1BQU0sR0FBRyxJQUFJLENBQUM7SUFDZCxXQUFXLEdBQUcsS0FBSyxDQUFDO0FBQ3RCLENBQUM7QUFFRDs7R0FFRztBQUNILFNBQWdCLFFBQVEsQ0FBQyxHQUFHLElBQWU7SUFDekMsSUFBSSxNQUFNLEVBQUUsS0FBSyxFQUFFLENBQUM7UUFDbEIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxhQUFhLEVBQUUsR0FBRyxJQUFJLENBQUMsQ0FBQztJQUN0QyxDQUFDO0FBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogVGhpbmtIaXZlIFNESyB2My4wIC0gQ29uZmlndXJhdGlvblxuICovXG5cbmltcG9ydCB0eXBlIHsgSW5pdE9wdGlvbnMsIEZyYW1ld29yayB9IGZyb20gJy4vdHlwZXMnO1xuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBDT05TVEFOVFNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0IGNvbnN0IFNES19WRVJTSU9OID0gJzQuMC4wJztcbmV4cG9ydCBjb25zdCBERUZBVUxUX0VORFBPSU5UID0gJ2h0dHBzOi8vYXBwLnRoaW5raGl2ZS5haSc7XG5leHBvcnQgY29uc3QgREVGQVVMVF9TRVJWSUNFX05BTUUgPSAnbXktYWktYWdlbnQnO1xuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBHTE9CQUwgQ09ORklHIFNUQVRFXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVzb2x2ZWRDb25maWcge1xuICBhcGlLZXk6IHN0cmluZztcbiAgYWdlbnRJZDogc3RyaW5nO1xuICBlbmRwb2ludDogc3RyaW5nO1xuICBzZXJ2aWNlTmFtZTogc3RyaW5nO1xuICBhdXRvSW5zdHJ1bWVudDogYm9vbGVhbjtcbiAgZnJhbWV3b3JrczogRnJhbWV3b3JrW107XG4gIGRlYnVnOiBib29sZWFuO1xufVxuXG5sZXQgY29uZmlnOiBSZXNvbHZlZENvbmZpZyB8IG51bGwgPSBudWxsO1xubGV0IGluaXRpYWxpemVkID0gZmFsc2U7XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIENPTkZJRyBGVU5DVElPTlNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuLyoqXG4gKiBJbml0aWFsaXplIGNvbmZpZ3VyYXRpb24gZnJvbSBvcHRpb25zXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpbml0Q29uZmlnKG9wdGlvbnM6IEluaXRPcHRpb25zID0ge30pOiBSZXNvbHZlZENvbmZpZyB7XG4gIGNvbnN0IGFwaUtleSA9IG9wdGlvbnMuYXBpS2V5IHx8IHByb2Nlc3MuZW52LlRISU5LSElWRV9BUElfS0VZIHx8ICcnO1xuICBjb25zdCBhZ2VudElkID0gb3B0aW9ucy5hZ2VudElkIHx8IHByb2Nlc3MuZW52LlRISU5LSElWRV9BR0VOVF9JRCB8fCAnJztcblxuICBpZiAoIWFwaUtleSAmJiAhYWdlbnRJZCkge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICdFaXRoZXIgYXBpS2V5IG9yIGFnZW50SWQgbXVzdCBiZSBwcm92aWRlZCAob3Igc2V0IFRISU5LSElWRV9BUElfS0VZIGVudiB2YXIpJ1xuICAgICk7XG4gIH1cblxuICBjb25maWcgPSB7XG4gICAgYXBpS2V5LFxuICAgIGFnZW50SWQsXG4gICAgZW5kcG9pbnQ6IG9wdGlvbnMuZW5kcG9pbnQgfHwgcHJvY2Vzcy5lbnYuVEhJTktISVZFX0VORFBPSU5UIHx8IERFRkFVTFRfRU5EUE9JTlQsXG4gICAgc2VydmljZU5hbWU6IG9wdGlvbnMuc2VydmljZU5hbWUgfHwgcHJvY2Vzcy5lbnYuVEhJTktISVZFX1NFUlZJQ0VfTkFNRSB8fCBERUZBVUxUX1NFUlZJQ0VfTkFNRSxcbiAgICBhdXRvSW5zdHJ1bWVudDogb3B0aW9ucy5hdXRvSW5zdHJ1bWVudCA/PyBmYWxzZSxcbiAgICBmcmFtZXdvcmtzOiBvcHRpb25zLmZyYW1ld29ya3MgfHwgWydsYW5nY2hhaW4nLCAnb3BlbmFpJ10sXG4gICAgZGVidWc6IG9wdGlvbnMuZGVidWcgPz8gZmFsc2UsXG4gIH07XG5cbiAgaW5pdGlhbGl6ZWQgPSB0cnVlO1xuICByZXR1cm4gY29uZmlnO1xufVxuXG4vKipcbiAqIEdldCBjdXJyZW50IGNvbmZpZ3VyYXRpb25cbiAqIEB0aHJvd3MgaWYgbm90IGluaXRpYWxpemVkXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRDb25maWcoKTogUmVzb2x2ZWRDb25maWcge1xuICBpZiAoIWNvbmZpZyB8fCAhaW5pdGlhbGl6ZWQpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ1RoaW5rSGl2ZSBTREsgbm90IGluaXRpYWxpemVkLiBDYWxsIGluaXQoKSBmaXJzdC4nKTtcbiAgfVxuICByZXR1cm4gY29uZmlnO1xufVxuXG4vKipcbiAqIENoZWNrIGlmIFNESyBpcyBpbml0aWFsaXplZFxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNJbml0aWFsaXplZCgpOiBib29sZWFuIHtcbiAgcmV0dXJuIGluaXRpYWxpemVkO1xufVxuXG4vKipcbiAqIFJlc2V0IGNvbmZpZ3VyYXRpb24gKGZvciB0ZXN0aW5nKVxuICovXG5leHBvcnQgZnVuY3Rpb24gcmVzZXRDb25maWcoKTogdm9pZCB7XG4gIGNvbmZpZyA9IG51bGw7XG4gIGluaXRpYWxpemVkID0gZmFsc2U7XG59XG5cbi8qKlxuICogRGVidWcgbG9nIGhlbHBlclxuICovXG5leHBvcnQgZnVuY3Rpb24gZGVidWdMb2coLi4uYXJnczogdW5rbm93bltdKTogdm9pZCB7XG4gIGlmIChjb25maWc/LmRlYnVnKSB7XG4gICAgY29uc29sZS5sb2coJ1tUaGlua0hpdmVdJywgLi4uYXJncyk7XG4gIH1cbn1cbiJdfQ==
@@ -18,8 +18,6 @@ export interface InitOptions {
18
18
  frameworks?: Framework[];
19
19
  /** Enable debug logging */
20
20
  debug?: boolean;
21
- /** API version (default: 'v3') */
22
- apiVersion?: 'v1' | 'v3';
23
21
  }
24
22
  export type Framework = 'langchain' | 'langgraph' | 'openai' | 'openai-assistants' | 'anthropic' | 'llamaindex' | 'n8n' | 'voiceflow';
25
23
  /**
@@ -50,6 +48,32 @@ export interface RunOptions {
50
48
  ticketLinking?: TicketLinkingOptions;
51
49
  /** Custom metadata */
52
50
  metadata?: Record<string, unknown>;
51
+ /**
52
+ * Custom business metrics to record with this run.
53
+ * Keys should match metrics declared during agent setup (onboarding).
54
+ * Values can be numbers, booleans, or short strings (max 100 chars).
55
+ *
56
+ * Metrics are aggregated over time for dashboard display and trend analysis.
57
+ * Undeclared metrics will be tracked but flagged as "discovered" metrics.
58
+ *
59
+ * @example
60
+ * metrics: {
61
+ * customer_satisfaction: 4.5,
62
+ * resolution_time_ms: 2450,
63
+ * escalated: false,
64
+ * ticket_priority: 'high'
65
+ * }
66
+ */
67
+ metrics?: Record<string, number | boolean | string>;
68
+ /**
69
+ * Request evaluation when run is ingested.
70
+ * Overrides agent's auto_evaluate setting for this run.
71
+ *
72
+ * - true: Force evaluation on this run
73
+ * - false: Skip evaluation even if agent has auto_evaluate enabled
74
+ * - undefined: Use agent's auto_evaluate setting (default)
75
+ */
76
+ runEvaluation?: boolean;
53
77
  }
54
78
  export type RunOutcome = 'resolved' | 'escalated' | 'abandoned' | 'failed' | 'transferred' | 'pending';
55
79
  export interface ConversationMessage {
@@ -93,6 +117,11 @@ export interface TicketLinkingOptions {
93
117
  customFieldValue?: string;
94
118
  }
95
119
  export type LinkMethod = 'sdk_explicit' | 'zendesk_marker' | 'custom_field' | 'middleware_stamp' | 'session_match' | 'email_time_window' | 'manual';
120
+ /**
121
+ * Classification flags for categorizing trace issues
122
+ * Separate from binary outcome (success/failure)
123
+ */
124
+ export type TraceCustomFlag = 'hallucination' | 'policy_violation' | 'tone_issue' | 'retrieval_miss' | 'error';
96
125
  /**
97
126
  * Trace options (v2 format, converted to runs internally)
98
127
  * @deprecated Use RunOptions instead
@@ -104,8 +133,10 @@ export interface TraceOptions {
104
133
  agentResponse: string;
105
134
  /** User's detected intent */
106
135
  userIntent?: string;
107
- /** Outcome: success, failure, partial_success */
108
- outcome?: 'success' | 'failure' | 'partial_success';
136
+ /** Binary outcome: success or failure */
137
+ outcome?: 'success' | 'failure';
138
+ /** Classification flags for categorizing the issue type (hallucination, policy_violation, etc.) */
139
+ customFlags?: TraceCustomFlag[];
109
140
  /** Duration in milliseconds */
110
141
  duration?: number;
111
142
  /** Session ID for conversation tracking */
@@ -121,6 +152,28 @@ export interface TraceOptions {
121
152
  businessContext?: BusinessContext;
122
153
  /** Custom metadata */
123
154
  metadata?: Record<string, unknown>;
155
+ /**
156
+ * Custom business metrics to record with this trace.
157
+ * Keys should match metrics declared during agent setup.
158
+ * Values can be numbers, booleans, or short strings.
159
+ *
160
+ * @example
161
+ * metrics: {
162
+ * customer_satisfaction: 4.5,
163
+ * resolution_time_ms: 2450,
164
+ * escalated: false
165
+ * }
166
+ */
167
+ metrics?: Record<string, number | boolean | string>;
168
+ /**
169
+ * Request evaluation when trace is ingested.
170
+ * Overrides agent's auto_evaluate setting for this trace.
171
+ *
172
+ * - true: Force evaluation on this trace
173
+ * - false: Skip evaluation even if agent has auto_evaluate enabled
174
+ * - undefined: Use agent's auto_evaluate setting (default)
175
+ */
176
+ runEvaluation?: boolean;
124
177
  }
125
178
  export interface SpanData {
126
179
  id?: string;
@@ -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}\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   * Custom business metrics to record with this run.\n   * Keys should match metrics declared during agent setup (onboarding).\n   * Values can be numbers, booleans, or short strings (max 100 chars).\n   *\n   * Metrics are aggregated over time for dashboard display and trend analysis.\n   * Undeclared metrics will be tracked but flagged as \"discovered\" metrics.\n   *\n   * @example\n   * metrics: {\n   *   customer_satisfaction: 4.5,\n   *   resolution_time_ms: 2450,\n   *   escalated: false,\n   *   ticket_priority: 'high'\n   * }\n   */\n  metrics?: Record<string, number | boolean | string>;\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   * Custom business metrics to record with this trace.\n   * Keys should match metrics declared during agent setup.\n   * Values can be numbers, booleans, or short strings.\n   *\n   * @example\n   * metrics: {\n   *   customer_satisfaction: 4.5,\n   *   resolution_time_ms: 2450,\n   *   escalated: false\n   * }\n   */\n  metrics?: Record<string, number | boolean | string>;\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"]}