@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.
- package/MIGRATION.md +83 -12
- package/README.md +279 -128
- package/dist/api/agents.d.ts +169 -0
- package/dist/api/agents.js +185 -0
- 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/calibration.d.ts +0 -62
- package/dist/api/calibration.js +5 -48
- package/dist/api/claims.js +10 -7
- 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/runs.js +12 -6
- package/dist/api/transcript-patterns.d.ts +204 -0
- package/dist/api/transcript-patterns.js +227 -0
- package/dist/core/client.d.ts +83 -9
- package/dist/core/client.js +229 -34
- package/dist/core/config.d.ts +2 -3
- package/dist/core/config.js +3 -4
- package/dist/core/types.d.ts +57 -4
- package/dist/core/types.js +1 -1
- package/dist/index.d.ts +429 -76
- package/dist/index.js +262 -42
- package/package.json +2 -2
|
@@ -0,0 +1,298 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* ThinkHive SDK - API Keys Management
|
|
4
|
+
*
|
|
5
|
+
* Create and manage scoped API keys with:
|
|
6
|
+
* - Permission control (read/write/delete)
|
|
7
|
+
* - Agent-level scoping (allowedAgentIds)
|
|
8
|
+
* - Environment separation (production/staging/development)
|
|
9
|
+
* - IP whitelisting (allowedIps)
|
|
10
|
+
*
|
|
11
|
+
* @example
|
|
12
|
+
* ```typescript
|
|
13
|
+
* import { apiKeys } from 'thinkhive-js';
|
|
14
|
+
*
|
|
15
|
+
* // Create a read-only key for monitoring
|
|
16
|
+
* const readOnlyKey = await apiKeys.create({
|
|
17
|
+
* name: 'Monitoring Key',
|
|
18
|
+
* scopeType: 'readonly'
|
|
19
|
+
* });
|
|
20
|
+
*
|
|
21
|
+
* // Create a key scoped to specific agents
|
|
22
|
+
* const agentKey = await apiKeys.create({
|
|
23
|
+
* name: 'Agent A Key',
|
|
24
|
+
* allowedAgentIds: ['agent-a-id', 'agent-b-id'],
|
|
25
|
+
* permissions: { read: true, write: true, delete: false }
|
|
26
|
+
* });
|
|
27
|
+
*
|
|
28
|
+
* // Create a staging environment key
|
|
29
|
+
* const stagingKey = await apiKeys.create({
|
|
30
|
+
* name: 'Staging Key',
|
|
31
|
+
* environment: 'staging',
|
|
32
|
+
* allowedIps: ['10.0.0.1', '10.0.0.2']
|
|
33
|
+
* });
|
|
34
|
+
* ```
|
|
35
|
+
*/
|
|
36
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
37
|
+
exports.apiKeys = void 0;
|
|
38
|
+
exports.create = create;
|
|
39
|
+
exports.list = list;
|
|
40
|
+
exports.revoke = revoke;
|
|
41
|
+
exports.rotate = rotate;
|
|
42
|
+
exports.test = test;
|
|
43
|
+
exports.hasPermission = hasPermission;
|
|
44
|
+
exports.isExpired = isExpired;
|
|
45
|
+
exports.isValid = isValid;
|
|
46
|
+
exports.getTimeUntilExpiry = getTimeUntilExpiry;
|
|
47
|
+
exports.canAccessAgent = canAccessAgent;
|
|
48
|
+
const client_1 = require("../core/client");
|
|
49
|
+
// ============================================================================
|
|
50
|
+
// API Key Management Functions
|
|
51
|
+
// ============================================================================
|
|
52
|
+
/**
|
|
53
|
+
* Create a new API key with specified permissions and scoping
|
|
54
|
+
*
|
|
55
|
+
* @param options - API key configuration
|
|
56
|
+
* @returns Created key with secret (only returned once!)
|
|
57
|
+
*
|
|
58
|
+
* @example
|
|
59
|
+
* ```typescript
|
|
60
|
+
* // Create a production write key
|
|
61
|
+
* const result = await apiKeys.create({
|
|
62
|
+
* name: 'Production SDK Key',
|
|
63
|
+
* permissions: { read: true, write: true, delete: false }
|
|
64
|
+
* });
|
|
65
|
+
*
|
|
66
|
+
* // Save the key securely - it won't be shown again!
|
|
67
|
+
* console.log('Save this key:', result.key);
|
|
68
|
+
* ```
|
|
69
|
+
*/
|
|
70
|
+
async function create(options) {
|
|
71
|
+
if (!options.name || options.name.trim().length === 0) {
|
|
72
|
+
throw new client_1.ThinkHiveValidationError('API key name is required', 'name');
|
|
73
|
+
}
|
|
74
|
+
// Validate IP addresses if provided
|
|
75
|
+
if (options.allowedIps) {
|
|
76
|
+
// IPv4 pattern
|
|
77
|
+
const ipv4Regex = /^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/;
|
|
78
|
+
// IPv6 pattern (supports full and compressed notation like ::1, 2001:db8::1)
|
|
79
|
+
const ipv6Regex = /^(?:(?:[0-9a-fA-F]{1,4}:){7}[0-9a-fA-F]{1,4}|(?:[0-9a-fA-F]{1,4}:){1,7}:|(?:[0-9a-fA-F]{1,4}:){1,6}:[0-9a-fA-F]{1,4}|(?:[0-9a-fA-F]{1,4}:){1,5}(?::[0-9a-fA-F]{1,4}){1,2}|(?:[0-9a-fA-F]{1,4}:){1,4}(?::[0-9a-fA-F]{1,4}){1,3}|(?:[0-9a-fA-F]{1,4}:){1,3}(?::[0-9a-fA-F]{1,4}){1,4}|(?:[0-9a-fA-F]{1,4}:){1,2}(?::[0-9a-fA-F]{1,4}){1,5}|[0-9a-fA-F]{1,4}:(?::[0-9a-fA-F]{1,4}){1,6}|:(?::[0-9a-fA-F]{1,4}){1,7}|::)$/;
|
|
80
|
+
for (const ip of options.allowedIps) {
|
|
81
|
+
if (!ipv4Regex.test(ip) && !ipv6Regex.test(ip) && ip !== 'localhost') {
|
|
82
|
+
throw new client_1.ThinkHiveValidationError(`Invalid IP address: ${ip}`, 'allowedIps');
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
const body = {
|
|
87
|
+
name: options.name.trim(),
|
|
88
|
+
};
|
|
89
|
+
if (options.permissions) {
|
|
90
|
+
body.permissions = {
|
|
91
|
+
read: options.permissions.read !== false,
|
|
92
|
+
write: options.permissions.write === true,
|
|
93
|
+
delete: options.permissions.delete === true,
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
if (options.scopeType) {
|
|
97
|
+
body.scopeType = options.scopeType;
|
|
98
|
+
}
|
|
99
|
+
if (options.environment) {
|
|
100
|
+
body.environment = options.environment;
|
|
101
|
+
}
|
|
102
|
+
if (options.allowedAgentIds && options.allowedAgentIds.length > 0) {
|
|
103
|
+
body.allowedAgentIds = options.allowedAgentIds;
|
|
104
|
+
}
|
|
105
|
+
if (options.allowedIps && options.allowedIps.length > 0) {
|
|
106
|
+
body.allowedIps = options.allowedIps;
|
|
107
|
+
}
|
|
108
|
+
if (options.expiresAt) {
|
|
109
|
+
body.expiresAt = options.expiresAt.toISOString();
|
|
110
|
+
}
|
|
111
|
+
return (0, client_1.apiRequest)('/api-keys', {
|
|
112
|
+
method: 'POST',
|
|
113
|
+
body,
|
|
114
|
+
apiVersion: 'v2',
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* List all API keys for the authenticated company
|
|
119
|
+
*
|
|
120
|
+
* @returns Array of API keys (without secrets)
|
|
121
|
+
*
|
|
122
|
+
* @example
|
|
123
|
+
* ```typescript
|
|
124
|
+
* const keys = await apiKeys.list();
|
|
125
|
+
* console.log(`Found ${keys.length} API keys`);
|
|
126
|
+
*
|
|
127
|
+
* // Filter by environment
|
|
128
|
+
* const prodKeys = keys.filter(k => k.environment === 'production');
|
|
129
|
+
* ```
|
|
130
|
+
*/
|
|
131
|
+
async function list() {
|
|
132
|
+
return (0, client_1.apiRequest)('/api-keys', {
|
|
133
|
+
method: 'GET',
|
|
134
|
+
apiVersion: 'v2',
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Revoke (deactivate) an API key
|
|
139
|
+
*
|
|
140
|
+
* @param keyId - ID of the key to revoke
|
|
141
|
+
*
|
|
142
|
+
* @example
|
|
143
|
+
* ```typescript
|
|
144
|
+
* // Revoke a compromised key
|
|
145
|
+
* await apiKeys.revoke('key-id-to-revoke');
|
|
146
|
+
* ```
|
|
147
|
+
*/
|
|
148
|
+
async function revoke(keyId) {
|
|
149
|
+
if (!keyId) {
|
|
150
|
+
throw new client_1.ThinkHiveValidationError('Key ID is required', 'keyId');
|
|
151
|
+
}
|
|
152
|
+
await (0, client_1.apiRequest)(`/api-keys/${keyId}`, {
|
|
153
|
+
method: 'DELETE',
|
|
154
|
+
apiVersion: 'v2',
|
|
155
|
+
});
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
158
|
+
* Rotate an API key (revoke old, create new with same config)
|
|
159
|
+
*
|
|
160
|
+
* @param keyId - ID of the key to rotate
|
|
161
|
+
* @param options - Optional overrides for the new key
|
|
162
|
+
* @returns New key with secret
|
|
163
|
+
*
|
|
164
|
+
* @example
|
|
165
|
+
* ```typescript
|
|
166
|
+
* // Rotate a key periodically
|
|
167
|
+
* const existingKeys = await apiKeys.list();
|
|
168
|
+
* const oldKey = existingKeys.find(k => k.name === 'Production Key');
|
|
169
|
+
*
|
|
170
|
+
* if (oldKey) {
|
|
171
|
+
* const newKey = await apiKeys.rotate(oldKey.id);
|
|
172
|
+
* // Update your systems with newKey.key
|
|
173
|
+
* console.log('New key:', newKey.key);
|
|
174
|
+
* }
|
|
175
|
+
* ```
|
|
176
|
+
*/
|
|
177
|
+
async function rotate(keyId, options) {
|
|
178
|
+
// Get existing key metadata
|
|
179
|
+
const keys = await list();
|
|
180
|
+
const existingKey = keys.find(k => k.id === keyId);
|
|
181
|
+
if (!existingKey) {
|
|
182
|
+
throw new client_1.ThinkHiveValidationError('API key not found', 'keyId');
|
|
183
|
+
}
|
|
184
|
+
// Create new key with same (or overridden) config
|
|
185
|
+
const newKeyOptions = {
|
|
186
|
+
name: options?.name || existingKey.name,
|
|
187
|
+
scopeType: options?.scopeType || existingKey.scopeType,
|
|
188
|
+
environment: options?.environment || existingKey.environment,
|
|
189
|
+
allowedAgentIds: options?.allowedAgentIds || existingKey.allowedAgentIds || undefined,
|
|
190
|
+
allowedIps: options?.allowedIps || existingKey.allowedIps || undefined,
|
|
191
|
+
};
|
|
192
|
+
// Handle permissions conversion
|
|
193
|
+
if (options?.permissions) {
|
|
194
|
+
newKeyOptions.permissions = options.permissions;
|
|
195
|
+
}
|
|
196
|
+
else if (existingKey.permissions && typeof existingKey.permissions === 'object' && !Array.isArray(existingKey.permissions)) {
|
|
197
|
+
newKeyOptions.permissions = existingKey.permissions;
|
|
198
|
+
}
|
|
199
|
+
// Create new key first (so we don't lose access if creation fails)
|
|
200
|
+
const newKey = await create(newKeyOptions);
|
|
201
|
+
// Revoke old key
|
|
202
|
+
await revoke(keyId);
|
|
203
|
+
return newKey;
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Test an API key connection
|
|
207
|
+
*
|
|
208
|
+
* @param apiKey - The full API key value to test
|
|
209
|
+
* @param agentId - Optional agent ID to test with
|
|
210
|
+
* @returns Test result
|
|
211
|
+
*
|
|
212
|
+
* @example
|
|
213
|
+
* ```typescript
|
|
214
|
+
* const result = await apiKeys.test('thk_...', 'my-agent-id');
|
|
215
|
+
* if (result.success) {
|
|
216
|
+
* console.log('Key is valid!');
|
|
217
|
+
* }
|
|
218
|
+
* ```
|
|
219
|
+
*/
|
|
220
|
+
async function test(apiKey, agentId) {
|
|
221
|
+
if (!apiKey) {
|
|
222
|
+
throw new client_1.ThinkHiveValidationError('API key is required', 'apiKey');
|
|
223
|
+
}
|
|
224
|
+
return (0, client_1.apiRequest)('/api-keys/test', {
|
|
225
|
+
method: 'POST',
|
|
226
|
+
body: { apiKey, agentId, createTestTrace: !!agentId },
|
|
227
|
+
apiVersion: 'v2',
|
|
228
|
+
});
|
|
229
|
+
}
|
|
230
|
+
// ============================================================================
|
|
231
|
+
// Utility Functions
|
|
232
|
+
// ============================================================================
|
|
233
|
+
/**
|
|
234
|
+
* Check if an API key has a specific permission
|
|
235
|
+
*/
|
|
236
|
+
function hasPermission(key, permission) {
|
|
237
|
+
// Handle object format
|
|
238
|
+
if (typeof key.permissions === 'object' && !Array.isArray(key.permissions)) {
|
|
239
|
+
return key.permissions[permission] === true;
|
|
240
|
+
}
|
|
241
|
+
// Handle legacy array format
|
|
242
|
+
if (Array.isArray(key.permissions)) {
|
|
243
|
+
return key.permissions.some(p => p.endsWith(`:${permission}`));
|
|
244
|
+
}
|
|
245
|
+
// Default: read allowed, write/delete denied
|
|
246
|
+
return permission === 'read';
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Check if an API key is expired
|
|
250
|
+
*/
|
|
251
|
+
function isExpired(key) {
|
|
252
|
+
if (!key.expiresAt)
|
|
253
|
+
return false;
|
|
254
|
+
return new Date(key.expiresAt) < new Date();
|
|
255
|
+
}
|
|
256
|
+
/**
|
|
257
|
+
* Check if an API key is valid (active and not expired)
|
|
258
|
+
*/
|
|
259
|
+
function isValid(key) {
|
|
260
|
+
return key.isActive && !isExpired(key);
|
|
261
|
+
}
|
|
262
|
+
/**
|
|
263
|
+
* Get time until key expires (in milliseconds)
|
|
264
|
+
* Returns null if key doesn't expire
|
|
265
|
+
*/
|
|
266
|
+
function getTimeUntilExpiry(key) {
|
|
267
|
+
if (!key.expiresAt)
|
|
268
|
+
return null;
|
|
269
|
+
const expiry = new Date(key.expiresAt).getTime();
|
|
270
|
+
const now = Date.now();
|
|
271
|
+
return Math.max(0, expiry - now);
|
|
272
|
+
}
|
|
273
|
+
/**
|
|
274
|
+
* Check if a key can access a specific agent
|
|
275
|
+
*/
|
|
276
|
+
function canAccessAgent(key, agentId) {
|
|
277
|
+
// If no agent restrictions, allow all
|
|
278
|
+
if (!key.allowedAgentIds || key.allowedAgentIds.length === 0) {
|
|
279
|
+
return true;
|
|
280
|
+
}
|
|
281
|
+
return key.allowedAgentIds.includes(agentId);
|
|
282
|
+
}
|
|
283
|
+
// ============================================================================
|
|
284
|
+
// Export namespace
|
|
285
|
+
// ============================================================================
|
|
286
|
+
exports.apiKeys = {
|
|
287
|
+
create,
|
|
288
|
+
list,
|
|
289
|
+
revoke,
|
|
290
|
+
rotate,
|
|
291
|
+
test,
|
|
292
|
+
hasPermission,
|
|
293
|
+
isExpired,
|
|
294
|
+
isValid,
|
|
295
|
+
getTimeUntilExpiry,
|
|
296
|
+
canAccessAgent,
|
|
297
|
+
};
|
|
298
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"apiKeys.js","sourceRoot":"","sources":["../../src/api/apiKeys.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ThinkHive SDK - API Keys Management\n *\n * Create and manage scoped API keys with:\n * - Permission control (read/write/delete)\n * - Agent-level scoping (allowedAgentIds)\n * - Environment separation (production/staging/development)\n * - IP whitelisting (allowedIps)\n *\n * @example\n * ```typescript\n * import { apiKeys } from 'thinkhive-js';\n *\n * // Create a read-only key for monitoring\n * const readOnlyKey = await apiKeys.create({\n *   name: 'Monitoring Key',\n *   scopeType: 'readonly'\n * });\n *\n * // Create a key scoped to specific agents\n * const agentKey = await apiKeys.create({\n *   name: 'Agent A Key',\n *   allowedAgentIds: ['agent-a-id', 'agent-b-id'],\n *   permissions: { read: true, write: true, delete: false }\n * });\n *\n * // Create a staging environment key\n * const stagingKey = await apiKeys.create({\n *   name: 'Staging Key',\n *   environment: 'staging',\n *   allowedIps: ['10.0.0.1', '10.0.0.2']\n * });\n * ```\n */\n\nimport { apiRequest, ThinkHiveValidationError } from '../core/client';\n\n// ============================================================================\n// Types\n// ============================================================================\n\n/**\n * API key permission configuration\n */\nexport interface ApiKeyPermissions {\n  /** Allow read operations (GET requests, listing, etc.) */\n  read: boolean;\n  /** Allow write operations (POST, PUT requests) */\n  write: boolean;\n  /** Allow delete operations (DELETE requests) */\n  delete: boolean;\n}\n\n/**\n * API key scope type\n * - `company`: Full access to all company data\n * - `agent`: Limited to specific agents (requires allowedAgentIds)\n * - `readonly`: Read-only access regardless of other settings\n */\nexport type ScopeType = 'company' | 'agent' | 'readonly';\n\n/**\n * API key environment\n * - `production`: For production systems\n * - `staging`: For staging/pre-production\n * - `development`: For local development\n */\nexport type Environment = 'production' | 'staging' | 'development';\n\n/**\n * Options for creating an API key\n */\nexport interface CreateApiKeyOptions {\n  /** Display name for the key */\n  name: string;\n\n  /** Permission configuration (defaults to read+write) */\n  permissions?: Partial<ApiKeyPermissions>;\n\n  /**\n   * Restrict key to specific agent IDs\n   * When set, key can only access traces for these agents\n   */\n  allowedAgentIds?: string[];\n\n  /**\n   * Scope type for the key\n   * @default 'company' or 'agent' if allowedAgentIds is set\n   */\n  scopeType?: ScopeType;\n\n  /**\n   * Environment for the key\n   * @default 'production'\n   */\n  environment?: Environment;\n\n  /**\n   * Restrict key to specific IP addresses\n   * When set, only requests from these IPs can use this key\n   */\n  allowedIps?: string[];\n\n  /** Expiration date for the key */\n  expiresAt?: Date;\n}\n\n/**\n * API key metadata (returned from server)\n */\nexport interface ApiKey {\n  /** Unique key ID */\n  id: string;\n  /** Display name */\n  name: string;\n  /** Key prefix (first 8 chars for identification) */\n  keyPrefix: string;\n  /** Permission configuration */\n  permissions: ApiKeyPermissions | string[];\n  /** Scope type */\n  scopeType: ScopeType;\n  /** Environment */\n  environment: Environment;\n  /** Allowed agent IDs (null = all agents) */\n  allowedAgentIds: string[] | null;\n  /** Allowed IP addresses (null = all IPs) */\n  allowedIps: string[] | null;\n  /** Whether key is active */\n  isActive: boolean;\n  /** Last usage timestamp */\n  lastUsedAt: string | null;\n  /** Total usage count */\n  usageCount: number;\n  /** Expiration date */\n  expiresAt: string | null;\n  /** Creation timestamp */\n  createdAt: string;\n}\n\n/**\n * Result of creating an API key (includes secret)\n */\nexport interface CreateApiKeyResult extends ApiKey {\n  /**\n   * The full API key value\n   * IMPORTANT: This is only returned once and cannot be retrieved later!\n   * Store it securely.\n   */\n  key: string;\n}\n\n// ============================================================================\n// API Key Management Functions\n// ============================================================================\n\n/**\n * Create a new API key with specified permissions and scoping\n *\n * @param options - API key configuration\n * @returns Created key with secret (only returned once!)\n *\n * @example\n * ```typescript\n * // Create a production write key\n * const result = await apiKeys.create({\n *   name: 'Production SDK Key',\n *   permissions: { read: true, write: true, delete: false }\n * });\n *\n * // Save the key securely - it won't be shown again!\n * console.log('Save this key:', result.key);\n * ```\n */\nexport async function create(options: CreateApiKeyOptions): Promise<CreateApiKeyResult> {\n  if (!options.name || options.name.trim().length === 0) {\n    throw new ThinkHiveValidationError('API key name is required', 'name');\n  }\n\n  // Validate IP addresses if provided\n  if (options.allowedIps) {\n    // IPv4 pattern\n    const ipv4Regex = /^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/;\n    // IPv6 pattern (supports full and compressed notation like ::1, 2001:db8::1)\n    const ipv6Regex = /^(?:(?:[0-9a-fA-F]{1,4}:){7}[0-9a-fA-F]{1,4}|(?:[0-9a-fA-F]{1,4}:){1,7}:|(?:[0-9a-fA-F]{1,4}:){1,6}:[0-9a-fA-F]{1,4}|(?:[0-9a-fA-F]{1,4}:){1,5}(?::[0-9a-fA-F]{1,4}){1,2}|(?:[0-9a-fA-F]{1,4}:){1,4}(?::[0-9a-fA-F]{1,4}){1,3}|(?:[0-9a-fA-F]{1,4}:){1,3}(?::[0-9a-fA-F]{1,4}){1,4}|(?:[0-9a-fA-F]{1,4}:){1,2}(?::[0-9a-fA-F]{1,4}){1,5}|[0-9a-fA-F]{1,4}:(?::[0-9a-fA-F]{1,4}){1,6}|:(?::[0-9a-fA-F]{1,4}){1,7}|::)$/;\n    for (const ip of options.allowedIps) {\n      if (!ipv4Regex.test(ip) && !ipv6Regex.test(ip) && ip !== 'localhost') {\n        throw new ThinkHiveValidationError(`Invalid IP address: ${ip}`, 'allowedIps');\n      }\n    }\n  }\n\n  const body: Record<string, unknown> = {\n    name: options.name.trim(),\n  };\n\n  if (options.permissions) {\n    body.permissions = {\n      read: options.permissions.read !== false,\n      write: options.permissions.write === true,\n      delete: options.permissions.delete === true,\n    };\n  }\n\n  if (options.scopeType) {\n    body.scopeType = options.scopeType;\n  }\n\n  if (options.environment) {\n    body.environment = options.environment;\n  }\n\n  if (options.allowedAgentIds && options.allowedAgentIds.length > 0) {\n    body.allowedAgentIds = options.allowedAgentIds;\n  }\n\n  if (options.allowedIps && options.allowedIps.length > 0) {\n    body.allowedIps = options.allowedIps;\n  }\n\n  if (options.expiresAt) {\n    body.expiresAt = options.expiresAt.toISOString();\n  }\n\n  return apiRequest<CreateApiKeyResult>('/api-keys', {\n    method: 'POST',\n    body,\n    apiVersion: 'v2',\n  });\n}\n\n/**\n * List all API keys for the authenticated company\n *\n * @returns Array of API keys (without secrets)\n *\n * @example\n * ```typescript\n * const keys = await apiKeys.list();\n * console.log(`Found ${keys.length} API keys`);\n *\n * // Filter by environment\n * const prodKeys = keys.filter(k => k.environment === 'production');\n * ```\n */\nexport async function list(): Promise<ApiKey[]> {\n  return apiRequest<ApiKey[]>('/api-keys', {\n    method: 'GET',\n    apiVersion: 'v2',\n  });\n}\n\n/**\n * Revoke (deactivate) an API key\n *\n * @param keyId - ID of the key to revoke\n *\n * @example\n * ```typescript\n * // Revoke a compromised key\n * await apiKeys.revoke('key-id-to-revoke');\n * ```\n */\nexport async function revoke(keyId: string): Promise<void> {\n  if (!keyId) {\n    throw new ThinkHiveValidationError('Key ID is required', 'keyId');\n  }\n\n  await apiRequest(`/api-keys/${keyId}`, {\n    method: 'DELETE',\n    apiVersion: 'v2',\n  });\n}\n\n/**\n * Rotate an API key (revoke old, create new with same config)\n *\n * @param keyId - ID of the key to rotate\n * @param options - Optional overrides for the new key\n * @returns New key with secret\n *\n * @example\n * ```typescript\n * // Rotate a key periodically\n * const existingKeys = await apiKeys.list();\n * const oldKey = existingKeys.find(k => k.name === 'Production Key');\n *\n * if (oldKey) {\n *   const newKey = await apiKeys.rotate(oldKey.id);\n *   // Update your systems with newKey.key\n *   console.log('New key:', newKey.key);\n * }\n * ```\n */\nexport async function rotate(\n  keyId: string,\n  options?: Partial<CreateApiKeyOptions>\n): Promise<CreateApiKeyResult> {\n  // Get existing key metadata\n  const keys = await list();\n  const existingKey = keys.find(k => k.id === keyId);\n\n  if (!existingKey) {\n    throw new ThinkHiveValidationError('API key not found', 'keyId');\n  }\n\n  // Create new key with same (or overridden) config\n  const newKeyOptions: CreateApiKeyOptions = {\n    name: options?.name || existingKey.name,\n    scopeType: options?.scopeType || existingKey.scopeType,\n    environment: options?.environment || existingKey.environment,\n    allowedAgentIds: options?.allowedAgentIds || existingKey.allowedAgentIds || undefined,\n    allowedIps: options?.allowedIps || existingKey.allowedIps || undefined,\n  };\n\n  // Handle permissions conversion\n  if (options?.permissions) {\n    newKeyOptions.permissions = options.permissions;\n  } else if (existingKey.permissions && typeof existingKey.permissions === 'object' && !Array.isArray(existingKey.permissions)) {\n    newKeyOptions.permissions = existingKey.permissions as ApiKeyPermissions;\n  }\n\n  // Create new key first (so we don't lose access if creation fails)\n  const newKey = await create(newKeyOptions);\n\n  // Revoke old key\n  await revoke(keyId);\n\n  return newKey;\n}\n\n/**\n * Test an API key connection\n *\n * @param apiKey - The full API key value to test\n * @param agentId - Optional agent ID to test with\n * @returns Test result\n *\n * @example\n * ```typescript\n * const result = await apiKeys.test('thk_...', 'my-agent-id');\n * if (result.success) {\n *   console.log('Key is valid!');\n * }\n * ```\n */\nexport async function test(\n  apiKey: string,\n  agentId?: string\n): Promise<{ success: boolean; error?: string; testTraceId?: string }> {\n  if (!apiKey) {\n    throw new ThinkHiveValidationError('API key is required', 'apiKey');\n  }\n\n  return apiRequest<{ success: boolean; error?: string; testTraceId?: string }>('/api-keys/test', {\n    method: 'POST',\n    body: { apiKey, agentId, createTestTrace: !!agentId },\n    apiVersion: 'v2',\n  });\n}\n\n// ============================================================================\n// Utility Functions\n// ============================================================================\n\n/**\n * Check if an API key has a specific permission\n */\nexport function hasPermission(\n  key: ApiKey,\n  permission: 'read' | 'write' | 'delete'\n): boolean {\n  // Handle object format\n  if (typeof key.permissions === 'object' && !Array.isArray(key.permissions)) {\n    return key.permissions[permission] === true;\n  }\n\n  // Handle legacy array format\n  if (Array.isArray(key.permissions)) {\n    return key.permissions.some(p => p.endsWith(`:${permission}`));\n  }\n\n  // Default: read allowed, write/delete denied\n  return permission === 'read';\n}\n\n/**\n * Check if an API key is expired\n */\nexport function isExpired(key: ApiKey): boolean {\n  if (!key.expiresAt) return false;\n  return new Date(key.expiresAt) < new Date();\n}\n\n/**\n * Check if an API key is valid (active and not expired)\n */\nexport function isValid(key: ApiKey): boolean {\n  return key.isActive && !isExpired(key);\n}\n\n/**\n * Get time until key expires (in milliseconds)\n * Returns null if key doesn't expire\n */\nexport function getTimeUntilExpiry(key: ApiKey): number | null {\n  if (!key.expiresAt) return null;\n  const expiry = new Date(key.expiresAt).getTime();\n  const now = Date.now();\n  return Math.max(0, expiry - now);\n}\n\n/**\n * Check if a key can access a specific agent\n */\nexport function canAccessAgent(key: ApiKey, agentId: string): boolean {\n  // If no agent restrictions, allow all\n  if (!key.allowedAgentIds || key.allowedAgentIds.length === 0) {\n    return true;\n  }\n  return key.allowedAgentIds.includes(agentId);\n}\n\n// ============================================================================\n// Export namespace\n// ============================================================================\n\nexport const apiKeys = {\n  create,\n  list,\n  revoke,\n  rotate,\n  test,\n  hasPermission,\n  isExpired,\n  isValid,\n  getTimeUntilExpiry,\n  canAccessAgent,\n};\n"]}
|
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ThinkHive SDK v3.2 - Business Metrics API
|
|
3
|
+
*
|
|
4
|
+
* Industry-driven business metrics with historical tracking and external data support
|
|
5
|
+
*/
|
|
6
|
+
/**
|
|
7
|
+
* Metric trend information
|
|
8
|
+
*/
|
|
9
|
+
export interface MetricTrend {
|
|
10
|
+
direction: 'up' | 'down' | 'stable';
|
|
11
|
+
value: number;
|
|
12
|
+
isPositive: boolean;
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* Current metric status
|
|
16
|
+
*/
|
|
17
|
+
export type MetricStatus = 'ready' | 'insufficient_data' | 'awaiting_external' | 'stale';
|
|
18
|
+
/**
|
|
19
|
+
* Current metric response from the API
|
|
20
|
+
*/
|
|
21
|
+
export interface CurrentMetricResponse {
|
|
22
|
+
metricName: string;
|
|
23
|
+
metricType: 'trace_calculated' | 'external';
|
|
24
|
+
value: number | null;
|
|
25
|
+
valueFormatted: string;
|
|
26
|
+
unit?: string;
|
|
27
|
+
status: MetricStatus;
|
|
28
|
+
statusMessage?: string;
|
|
29
|
+
trend?: MetricTrend;
|
|
30
|
+
traceCount?: number;
|
|
31
|
+
minTraceThreshold?: number;
|
|
32
|
+
progressPercent?: number;
|
|
33
|
+
lastExternalUpdate?: string;
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Historical data point
|
|
37
|
+
*/
|
|
38
|
+
export interface MetricHistoryPoint {
|
|
39
|
+
periodStart: string;
|
|
40
|
+
periodEnd: string;
|
|
41
|
+
value: number;
|
|
42
|
+
valueFormatted?: string;
|
|
43
|
+
source: string;
|
|
44
|
+
traceCount?: number;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Historical summary statistics
|
|
48
|
+
*/
|
|
49
|
+
export interface MetricHistorySummary {
|
|
50
|
+
current: number | null;
|
|
51
|
+
previous: number | null;
|
|
52
|
+
changePercent: number | null;
|
|
53
|
+
isPositive: boolean;
|
|
54
|
+
dataPointCount: number;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Full history response
|
|
58
|
+
*/
|
|
59
|
+
export interface MetricHistoryResponse {
|
|
60
|
+
metricName: string;
|
|
61
|
+
unit: string;
|
|
62
|
+
higherIsBetter: boolean;
|
|
63
|
+
dataPoints: MetricHistoryPoint[];
|
|
64
|
+
summary: MetricHistorySummary;
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Options for recording external metric values
|
|
68
|
+
*/
|
|
69
|
+
export interface RecordMetricOptions {
|
|
70
|
+
metricName: string;
|
|
71
|
+
value: number;
|
|
72
|
+
unit?: string;
|
|
73
|
+
periodStart: string | Date;
|
|
74
|
+
periodEnd: string | Date;
|
|
75
|
+
source?: string;
|
|
76
|
+
sourceDetails?: Record<string, unknown>;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Response from recording a metric value
|
|
80
|
+
*/
|
|
81
|
+
export interface RecordMetricResponse {
|
|
82
|
+
success: boolean;
|
|
83
|
+
id: string;
|
|
84
|
+
recordedAt: string;
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Business Metrics API client for industry-driven metrics with historical tracking
|
|
88
|
+
*/
|
|
89
|
+
export declare const businessMetrics: {
|
|
90
|
+
/**
|
|
91
|
+
* Get current metric value with status information
|
|
92
|
+
*
|
|
93
|
+
* @example
|
|
94
|
+
* ```typescript
|
|
95
|
+
* const metric = await businessMetrics.current('agent_123', 'Deflection Rate');
|
|
96
|
+
* console.log(`${metric.metricName}: ${metric.valueFormatted}`);
|
|
97
|
+
*
|
|
98
|
+
* if (metric.status === 'insufficient_data') {
|
|
99
|
+
* console.log(`Need ${metric.minTraceThreshold - metric.traceCount} more traces`);
|
|
100
|
+
* }
|
|
101
|
+
* ```
|
|
102
|
+
*/
|
|
103
|
+
current(agentId: string, metricName?: string): Promise<CurrentMetricResponse>;
|
|
104
|
+
/**
|
|
105
|
+
* Get historical metric data for graphing
|
|
106
|
+
*
|
|
107
|
+
* @example
|
|
108
|
+
* ```typescript
|
|
109
|
+
* const history = await businessMetrics.history('agent_123', 'Deflection Rate', {
|
|
110
|
+
* startDate: new Date(Date.now() - 30 * 24 * 60 * 60 * 1000),
|
|
111
|
+
* endDate: new Date(),
|
|
112
|
+
* granularity: 'daily',
|
|
113
|
+
* });
|
|
114
|
+
*
|
|
115
|
+
* console.log(`${history.dataPoints.length} data points`);
|
|
116
|
+
* console.log(`Change: ${history.summary.changePercent}%`);
|
|
117
|
+
* ```
|
|
118
|
+
*/
|
|
119
|
+
history(agentId: string, metricName: string, options?: {
|
|
120
|
+
startDate?: string | Date;
|
|
121
|
+
endDate?: string | Date;
|
|
122
|
+
granularity?: "hourly" | "daily" | "weekly" | "monthly";
|
|
123
|
+
}): Promise<MetricHistoryResponse>;
|
|
124
|
+
/**
|
|
125
|
+
* Record an external metric value
|
|
126
|
+
*
|
|
127
|
+
* Use this to ingest metrics from external data sources (CRM, surveys, billing, etc.)
|
|
128
|
+
*
|
|
129
|
+
* @example
|
|
130
|
+
* ```typescript
|
|
131
|
+
* // Record CSAT score from survey system
|
|
132
|
+
* await businessMetrics.record('agent_123', {
|
|
133
|
+
* metricName: 'CSAT/NPS',
|
|
134
|
+
* value: 4.5,
|
|
135
|
+
* unit: 'score',
|
|
136
|
+
* periodStart: '2024-01-01T00:00:00Z',
|
|
137
|
+
* periodEnd: '2024-01-07T23:59:59Z',
|
|
138
|
+
* source: 'survey_system',
|
|
139
|
+
* sourceDetails: { surveyId: 'survey_456', responseCount: 150 },
|
|
140
|
+
* });
|
|
141
|
+
* ```
|
|
142
|
+
*/
|
|
143
|
+
record(agentId: string, options: RecordMetricOptions): Promise<RecordMetricResponse>;
|
|
144
|
+
/**
|
|
145
|
+
* Batch record multiple external metric values
|
|
146
|
+
*
|
|
147
|
+
* @example
|
|
148
|
+
* ```typescript
|
|
149
|
+
* await businessMetrics.recordBatch('agent_123', [
|
|
150
|
+
* { metricName: 'CSAT/NPS', value: 4.5, ... },
|
|
151
|
+
* { metricName: 'Hours Saved', value: 120, ... },
|
|
152
|
+
* ]);
|
|
153
|
+
* ```
|
|
154
|
+
*/
|
|
155
|
+
recordBatch(agentId: string, metrics: RecordMetricOptions[]): Promise<RecordMetricResponse[]>;
|
|
156
|
+
};
|
|
157
|
+
/**
|
|
158
|
+
* Check if a metric is ready to display
|
|
159
|
+
*/
|
|
160
|
+
export declare function isMetricReady(metric: CurrentMetricResponse): boolean;
|
|
161
|
+
/**
|
|
162
|
+
* Check if a metric needs more trace data
|
|
163
|
+
*/
|
|
164
|
+
export declare function needsMoreTraces(metric: CurrentMetricResponse): boolean;
|
|
165
|
+
/**
|
|
166
|
+
* Check if a metric is waiting for external data
|
|
167
|
+
*/
|
|
168
|
+
export declare function awaitingExternalData(metric: CurrentMetricResponse): boolean;
|
|
169
|
+
/**
|
|
170
|
+
* Check if metric data is stale
|
|
171
|
+
*/
|
|
172
|
+
export declare function isMetricStale(metric: CurrentMetricResponse): boolean;
|
|
173
|
+
/**
|
|
174
|
+
* Get human-readable status message
|
|
175
|
+
*/
|
|
176
|
+
export declare function getStatusMessage(metric: CurrentMetricResponse): string;
|
|
177
|
+
/**
|
|
178
|
+
* Calculate progress toward minimum trace threshold
|
|
179
|
+
*/
|
|
180
|
+
export declare function getTraceProgress(metric: CurrentMetricResponse): number;
|
|
181
|
+
/**
|
|
182
|
+
* Format metric value for display based on unit
|
|
183
|
+
*/
|
|
184
|
+
export declare function formatMetricValue(value: number, unit: string): string;
|
|
185
|
+
/**
|
|
186
|
+
* Get trend direction as emoji
|
|
187
|
+
*/
|
|
188
|
+
export declare function getTrendEmoji(trend: MetricTrend): string;
|