@thinkhive/sdk 2.0.1 → 3.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,274 @@
1
+ "use strict";
2
+ /**
3
+ * ThinkHive SDK v3.0 - Customer Context
4
+ *
5
+ * Time-series customer metrics snapshots
6
+ * Captures ARR, health score, segment AS OF the run time (not current values)
7
+ */
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.customerContext = void 0;
10
+ exports.toContextSnapshot = toContextSnapshot;
11
+ exports.captureCustomerContext = captureCustomerContext;
12
+ exports.getContextAsOf = getContextAsOf;
13
+ exports.calculateArrChange = calculateArrChange;
14
+ exports.calculateHealthTrend = calculateHealthTrend;
15
+ const client_1 = require("../core/client");
16
+ // ============================================================================
17
+ // CUSTOMER CONTEXT API
18
+ // ============================================================================
19
+ /**
20
+ * Customer context API client for time-series metrics
21
+ */
22
+ exports.customerContext = {
23
+ /**
24
+ * Create a customer account
25
+ *
26
+ * @example
27
+ * ```typescript
28
+ * const account = await customerContext.createAccount({
29
+ * name: 'Acme Corp',
30
+ * externalId: 'sf_001234',
31
+ * externalSource: 'salesforce',
32
+ * segment: 'enterprise',
33
+ * });
34
+ * ```
35
+ */
36
+ async createAccount(input) {
37
+ return (0, client_1.apiRequestWithData)('/customers', {
38
+ method: 'POST',
39
+ body: input,
40
+ });
41
+ },
42
+ /**
43
+ * Get a customer account by ID
44
+ *
45
+ * @example
46
+ * ```typescript
47
+ * const account = await customerContext.getAccount('cust_abc123');
48
+ * ```
49
+ */
50
+ async getAccount(customerId) {
51
+ return (0, client_1.apiRequestWithData)(`/customers/${customerId}`);
52
+ },
53
+ /**
54
+ * Get a customer account by external ID
55
+ *
56
+ * @example
57
+ * ```typescript
58
+ * const account = await customerContext.getAccountByExternalId(
59
+ * 'sf_001234',
60
+ * 'salesforce'
61
+ * );
62
+ * ```
63
+ */
64
+ async getAccountByExternalId(externalId, source) {
65
+ try {
66
+ return await (0, client_1.apiRequestWithData)(`/customers/external/${source}/${externalId}`);
67
+ }
68
+ catch (error) {
69
+ if (error?.statusCode === 404) {
70
+ return null;
71
+ }
72
+ throw error;
73
+ }
74
+ },
75
+ /**
76
+ * Capture a metrics snapshot for a customer
77
+ * This creates a point-in-time record of the customer's metrics
78
+ *
79
+ * @example
80
+ * ```typescript
81
+ * // Capture current metrics
82
+ * const snapshot = await customerContext.captureSnapshot('cust_abc123', {
83
+ * arr: 120000,
84
+ * healthScore: 85,
85
+ * nps: 45,
86
+ * segment: 'enterprise',
87
+ * });
88
+ *
89
+ * // Use this snapshot in a run
90
+ * const run = await runs.create({
91
+ * agentId: 'agent_123',
92
+ * customerContext: {
93
+ * customerId: 'cust_abc123',
94
+ * arr: snapshot.arr,
95
+ * healthScore: snapshot.healthScore,
96
+ * capturedAt: snapshot.capturedAt,
97
+ * },
98
+ * // ...
99
+ * });
100
+ * ```
101
+ */
102
+ async captureSnapshot(customerId, metrics) {
103
+ return (0, client_1.apiRequestWithData)(`/customers/${customerId}/snapshots`, {
104
+ method: 'POST',
105
+ body: {
106
+ ...metrics,
107
+ capturedAt: new Date().toISOString(),
108
+ },
109
+ });
110
+ },
111
+ /**
112
+ * Get metrics snapshots for a customer (time-series)
113
+ *
114
+ * @example
115
+ * ```typescript
116
+ * // Get last 30 days of snapshots
117
+ * const snapshots = await customerContext.getSnapshots('cust_abc123', {
118
+ * from: new Date(Date.now() - 30 * 24 * 60 * 60 * 1000).toISOString(),
119
+ * });
120
+ * ```
121
+ */
122
+ async getSnapshots(customerId, options = {}) {
123
+ const params = new URLSearchParams();
124
+ if (options.from)
125
+ params.set('from', options.from);
126
+ if (options.to)
127
+ params.set('to', options.to);
128
+ if (options.limit)
129
+ params.set('limit', String(options.limit));
130
+ return (0, client_1.apiRequestWithData)(`/customers/${customerId}/snapshots?${params.toString()}`);
131
+ },
132
+ /**
133
+ * Get the most recent snapshot for a customer
134
+ *
135
+ * @example
136
+ * ```typescript
137
+ * const latest = await customerContext.getLatestSnapshot('cust_abc123');
138
+ * ```
139
+ */
140
+ async getLatestSnapshot(customerId) {
141
+ try {
142
+ return await (0, client_1.apiRequestWithData)(`/customers/${customerId}/snapshots/latest`);
143
+ }
144
+ catch (error) {
145
+ if (error?.statusCode === 404) {
146
+ return null;
147
+ }
148
+ throw error;
149
+ }
150
+ },
151
+ /**
152
+ * Get snapshot closest to a specific timestamp
153
+ * Useful for retroactive analysis
154
+ *
155
+ * @example
156
+ * ```typescript
157
+ * // Get metrics as of a specific date
158
+ * const snapshot = await customerContext.getSnapshotAsOf(
159
+ * 'cust_abc123',
160
+ * '2024-01-15T10:00:00Z'
161
+ * );
162
+ * ```
163
+ */
164
+ async getSnapshotAsOf(customerId, timestamp) {
165
+ const ts = new Date(timestamp).toISOString();
166
+ try {
167
+ return await (0, client_1.apiRequestWithData)(`/customers/${customerId}/snapshots/as-of?timestamp=${encodeURIComponent(ts)}`);
168
+ }
169
+ catch (error) {
170
+ if (error?.statusCode === 404) {
171
+ return null;
172
+ }
173
+ throw error;
174
+ }
175
+ },
176
+ };
177
+ // ============================================================================
178
+ // HELPER FUNCTIONS
179
+ // ============================================================================
180
+ /**
181
+ * Create a CustomerContextSnapshot from a metrics snapshot
182
+ */
183
+ function toContextSnapshot(snapshot) {
184
+ return {
185
+ customerId: snapshot.customerAccountId,
186
+ arr: snapshot.arr,
187
+ healthScore: snapshot.healthScore,
188
+ segment: snapshot.segment,
189
+ capturedAt: snapshot.capturedAt,
190
+ };
191
+ }
192
+ /**
193
+ * Capture metrics and create context snapshot in one call
194
+ *
195
+ * @example
196
+ * ```typescript
197
+ * const context = await captureCustomerContext('cust_abc123', {
198
+ * arr: 100000,
199
+ * healthScore: 90,
200
+ * segment: 'enterprise',
201
+ * });
202
+ *
203
+ * // Use in run
204
+ * const run = await runs.create({
205
+ * agentId: 'agent_123',
206
+ * customerContext: context,
207
+ * // ...
208
+ * });
209
+ * ```
210
+ */
211
+ async function captureCustomerContext(customerId, metrics) {
212
+ const snapshot = await exports.customerContext.captureSnapshot(customerId, metrics);
213
+ return toContextSnapshot(snapshot);
214
+ }
215
+ /**
216
+ * Get customer context as of a specific time
217
+ */
218
+ async function getContextAsOf(customerId, timestamp) {
219
+ const snapshot = await exports.customerContext.getSnapshotAsOf(customerId, timestamp);
220
+ return snapshot ? toContextSnapshot(snapshot) : null;
221
+ }
222
+ /**
223
+ * Calculate ARR change between two snapshots
224
+ */
225
+ function calculateArrChange(older, newer) {
226
+ const oldArr = older.arr || 0;
227
+ const newArr = newer.arr || 0;
228
+ const absolute = newArr - oldArr;
229
+ const percentage = oldArr > 0 ? (absolute / oldArr) * 100 : 0;
230
+ return {
231
+ absolute,
232
+ percentage,
233
+ direction: absolute > 0 ? 'increase' : absolute < 0 ? 'decrease' : 'stable',
234
+ };
235
+ }
236
+ /**
237
+ * Calculate health score trend
238
+ */
239
+ function calculateHealthTrend(snapshots) {
240
+ const scores = snapshots
241
+ .filter((s) => s.healthScore !== undefined && s.healthScore !== null)
242
+ .map((s) => s.healthScore);
243
+ if (scores.length === 0) {
244
+ return {
245
+ currentScore: null,
246
+ avgScore: null,
247
+ trend: 'stable',
248
+ volatility: 'low',
249
+ };
250
+ }
251
+ const currentScore = scores[0];
252
+ const avgScore = scores.reduce((a, b) => a + b, 0) / scores.length;
253
+ // Calculate trend (simple linear regression direction)
254
+ let trend = 'stable';
255
+ if (scores.length >= 2) {
256
+ const recentAvg = scores.slice(0, Math.ceil(scores.length / 2)).reduce((a, b) => a + b, 0) / Math.ceil(scores.length / 2);
257
+ const olderAvg = scores.slice(Math.ceil(scores.length / 2)).reduce((a, b) => a + b, 0) / Math.floor(scores.length / 2);
258
+ if (recentAvg - olderAvg > 5)
259
+ trend = 'improving';
260
+ else if (olderAvg - recentAvg > 5)
261
+ trend = 'declining';
262
+ }
263
+ // Calculate volatility (standard deviation)
264
+ const variance = scores.reduce((acc, s) => acc + Math.pow(s - avgScore, 2), 0) / scores.length;
265
+ const stdDev = Math.sqrt(variance);
266
+ const volatility = stdDev < 5 ? 'low' : stdDev < 15 ? 'medium' : 'high';
267
+ return {
268
+ currentScore,
269
+ avgScore,
270
+ trend,
271
+ volatility,
272
+ };
273
+ }
274
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"customer-context.js","sourceRoot":"","sources":["../../src/integrations/customer-context.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ThinkHive SDK v3.0 - Customer Context\n *\n * Time-series customer metrics snapshots\n * Captures ARR, health score, segment AS OF the run time (not current values)\n */\n\nimport { apiRequest, apiRequestWithData } from '../core/client';\nimport type { CustomerContextSnapshot } from '../core/types';\n\n// ============================================================================\n// CUSTOMER ACCOUNT TYPES\n// ============================================================================\n\nexport interface CustomerAccount {\n  id: string;\n  companyId: string;\n  externalId?: string;\n  externalSource?: 'salesforce' | 'hubspot' | 'zendesk' | 'intercom' | 'custom';\n  name: string;\n  domain?: string;\n  segment?: string;\n  industry?: string;\n  employeeCount?: number;\n  createdAt: string;\n  updatedAt: string;\n}\n\nexport interface CustomerMetricsSnapshot {\n  id: string;\n  customerAccountId: string;\n  arr?: number;\n  healthScore?: number;\n  nps?: number;\n  segment?: string;\n  churnRisk?: 'low' | 'medium' | 'high';\n  capturedAt: string;\n  source?: string;\n  createdAt: string;\n}\n\n// ============================================================================\n// CUSTOMER CONTEXT API\n// ============================================================================\n\n/**\n * Customer context API client for time-series metrics\n */\nexport const customerContext = {\n  /**\n   * Create a customer account\n   *\n   * @example\n   * ```typescript\n   * const account = await customerContext.createAccount({\n   *   name: 'Acme Corp',\n   *   externalId: 'sf_001234',\n   *   externalSource: 'salesforce',\n   *   segment: 'enterprise',\n   * });\n   * ```\n   */\n  async createAccount(input: {\n    name: string;\n    externalId?: string;\n    externalSource?: CustomerAccount['externalSource'];\n    domain?: string;\n    segment?: string;\n    industry?: string;\n    employeeCount?: number;\n  }): Promise<CustomerAccount> {\n    return apiRequestWithData<CustomerAccount>('/customers', {\n      method: 'POST',\n      body: input,\n    });\n  },\n\n  /**\n   * Get a customer account by ID\n   *\n   * @example\n   * ```typescript\n   * const account = await customerContext.getAccount('cust_abc123');\n   * ```\n   */\n  async getAccount(customerId: string): Promise<CustomerAccount> {\n    return apiRequestWithData<CustomerAccount>(`/customers/${customerId}`);\n  },\n\n  /**\n   * Get a customer account by external ID\n   *\n   * @example\n   * ```typescript\n   * const account = await customerContext.getAccountByExternalId(\n   *   'sf_001234',\n   *   'salesforce'\n   * );\n   * ```\n   */\n  async getAccountByExternalId(\n    externalId: string,\n    source: CustomerAccount['externalSource']\n  ): Promise<CustomerAccount | null> {\n    try {\n      return await apiRequestWithData<CustomerAccount>(\n        `/customers/external/${source}/${externalId}`\n      );\n    } catch (error) {\n      if ((error as any)?.statusCode === 404) {\n        return null;\n      }\n      throw error;\n    }\n  },\n\n  /**\n   * Capture a metrics snapshot for a customer\n   * This creates a point-in-time record of the customer's metrics\n   *\n   * @example\n   * ```typescript\n   * // Capture current metrics\n   * const snapshot = await customerContext.captureSnapshot('cust_abc123', {\n   *   arr: 120000,\n   *   healthScore: 85,\n   *   nps: 45,\n   *   segment: 'enterprise',\n   * });\n   *\n   * // Use this snapshot in a run\n   * const run = await runs.create({\n   *   agentId: 'agent_123',\n   *   customerContext: {\n   *     customerId: 'cust_abc123',\n   *     arr: snapshot.arr,\n   *     healthScore: snapshot.healthScore,\n   *     capturedAt: snapshot.capturedAt,\n   *   },\n   *   // ...\n   * });\n   * ```\n   */\n  async captureSnapshot(\n    customerId: string,\n    metrics: {\n      arr?: number;\n      healthScore?: number;\n      nps?: number;\n      segment?: string;\n      churnRisk?: 'low' | 'medium' | 'high';\n      source?: string;\n    }\n  ): Promise<CustomerMetricsSnapshot> {\n    return apiRequestWithData<CustomerMetricsSnapshot>(\n      `/customers/${customerId}/snapshots`,\n      {\n        method: 'POST',\n        body: {\n          ...metrics,\n          capturedAt: new Date().toISOString(),\n        },\n      }\n    );\n  },\n\n  /**\n   * Get metrics snapshots for a customer (time-series)\n   *\n   * @example\n   * ```typescript\n   * // Get last 30 days of snapshots\n   * const snapshots = await customerContext.getSnapshots('cust_abc123', {\n   *   from: new Date(Date.now() - 30 * 24 * 60 * 60 * 1000).toISOString(),\n   * });\n   * ```\n   */\n  async getSnapshots(\n    customerId: string,\n    options: {\n      from?: string;\n      to?: string;\n      limit?: number;\n    } = {}\n  ): Promise<CustomerMetricsSnapshot[]> {\n    const params = new URLSearchParams();\n    if (options.from) params.set('from', options.from);\n    if (options.to) params.set('to', options.to);\n    if (options.limit) params.set('limit', String(options.limit));\n\n    return apiRequestWithData<CustomerMetricsSnapshot[]>(\n      `/customers/${customerId}/snapshots?${params.toString()}`\n    );\n  },\n\n  /**\n   * Get the most recent snapshot for a customer\n   *\n   * @example\n   * ```typescript\n   * const latest = await customerContext.getLatestSnapshot('cust_abc123');\n   * ```\n   */\n  async getLatestSnapshot(\n    customerId: string\n  ): Promise<CustomerMetricsSnapshot | null> {\n    try {\n      return await apiRequestWithData<CustomerMetricsSnapshot>(\n        `/customers/${customerId}/snapshots/latest`\n      );\n    } catch (error) {\n      if ((error as any)?.statusCode === 404) {\n        return null;\n      }\n      throw error;\n    }\n  },\n\n  /**\n   * Get snapshot closest to a specific timestamp\n   * Useful for retroactive analysis\n   *\n   * @example\n   * ```typescript\n   * // Get metrics as of a specific date\n   * const snapshot = await customerContext.getSnapshotAsOf(\n   *   'cust_abc123',\n   *   '2024-01-15T10:00:00Z'\n   * );\n   * ```\n   */\n  async getSnapshotAsOf(\n    customerId: string,\n    timestamp: string | Date\n  ): Promise<CustomerMetricsSnapshot | null> {\n    const ts = new Date(timestamp).toISOString();\n    try {\n      return await apiRequestWithData<CustomerMetricsSnapshot>(\n        `/customers/${customerId}/snapshots/as-of?timestamp=${encodeURIComponent(ts)}`\n      );\n    } catch (error) {\n      if ((error as any)?.statusCode === 404) {\n        return null;\n      }\n      throw error;\n    }\n  },\n};\n\n// ============================================================================\n// HELPER FUNCTIONS\n// ============================================================================\n\n/**\n * Create a CustomerContextSnapshot from a metrics snapshot\n */\nexport function toContextSnapshot(\n  snapshot: CustomerMetricsSnapshot\n): CustomerContextSnapshot {\n  return {\n    customerId: snapshot.customerAccountId,\n    arr: snapshot.arr,\n    healthScore: snapshot.healthScore,\n    segment: snapshot.segment,\n    capturedAt: snapshot.capturedAt,\n  };\n}\n\n/**\n * Capture metrics and create context snapshot in one call\n *\n * @example\n * ```typescript\n * const context = await captureCustomerContext('cust_abc123', {\n *   arr: 100000,\n *   healthScore: 90,\n *   segment: 'enterprise',\n * });\n *\n * // Use in run\n * const run = await runs.create({\n *   agentId: 'agent_123',\n *   customerContext: context,\n *   // ...\n * });\n * ```\n */\nexport async function captureCustomerContext(\n  customerId: string,\n  metrics: {\n    arr?: number;\n    healthScore?: number;\n    segment?: string;\n  }\n): Promise<CustomerContextSnapshot> {\n  const snapshot = await customerContext.captureSnapshot(customerId, metrics);\n  return toContextSnapshot(snapshot);\n}\n\n/**\n * Get customer context as of a specific time\n */\nexport async function getContextAsOf(\n  customerId: string,\n  timestamp: string | Date\n): Promise<CustomerContextSnapshot | null> {\n  const snapshot = await customerContext.getSnapshotAsOf(customerId, timestamp);\n  return snapshot ? toContextSnapshot(snapshot) : null;\n}\n\n/**\n * Calculate ARR change between two snapshots\n */\nexport function calculateArrChange(\n  older: CustomerMetricsSnapshot,\n  newer: CustomerMetricsSnapshot\n): {\n  absolute: number;\n  percentage: number;\n  direction: 'increase' | 'decrease' | 'stable';\n} {\n  const oldArr = older.arr || 0;\n  const newArr = newer.arr || 0;\n  const absolute = newArr - oldArr;\n  const percentage = oldArr > 0 ? (absolute / oldArr) * 100 : 0;\n\n  return {\n    absolute,\n    percentage,\n    direction: absolute > 0 ? 'increase' : absolute < 0 ? 'decrease' : 'stable',\n  };\n}\n\n/**\n * Calculate health score trend\n */\nexport function calculateHealthTrend(\n  snapshots: CustomerMetricsSnapshot[]\n): {\n  currentScore: number | null;\n  avgScore: number | null;\n  trend: 'improving' | 'declining' | 'stable';\n  volatility: 'low' | 'medium' | 'high';\n} {\n  const scores = snapshots\n    .filter((s) => s.healthScore !== undefined && s.healthScore !== null)\n    .map((s) => s.healthScore!);\n\n  if (scores.length === 0) {\n    return {\n      currentScore: null,\n      avgScore: null,\n      trend: 'stable',\n      volatility: 'low',\n    };\n  }\n\n  const currentScore = scores[0];\n  const avgScore = scores.reduce((a, b) => a + b, 0) / scores.length;\n\n  // Calculate trend (simple linear regression direction)\n  let trend: 'improving' | 'declining' | 'stable' = 'stable';\n  if (scores.length >= 2) {\n    const recentAvg = scores.slice(0, Math.ceil(scores.length / 2)).reduce((a, b) => a + b, 0) / Math.ceil(scores.length / 2);\n    const olderAvg = scores.slice(Math.ceil(scores.length / 2)).reduce((a, b) => a + b, 0) / Math.floor(scores.length / 2);\n    if (recentAvg - olderAvg > 5) trend = 'improving';\n    else if (olderAvg - recentAvg > 5) trend = 'declining';\n  }\n\n  // Calculate volatility (standard deviation)\n  const variance = scores.reduce((acc, s) => acc + Math.pow(s - avgScore, 2), 0) / scores.length;\n  const stdDev = Math.sqrt(variance);\n  const volatility = stdDev < 5 ? 'low' : stdDev < 15 ? 'medium' : 'high';\n\n  return {\n    currentScore,\n    avgScore,\n    trend,\n    volatility,\n  };\n}\n"]}
@@ -0,0 +1,217 @@
1
+ /**
2
+ * ThinkHive SDK v3.0 - Ticket Linking
3
+ *
4
+ * Deterministic linking between runs and support tickets
5
+ * 7 link methods with explicit confidence scores
6
+ */
7
+ import type { LinkMethod } from '../core/types';
8
+ /**
9
+ * Generate a Zendesk marker to embed in agent responses
10
+ *
11
+ * @example
12
+ * ```typescript
13
+ * const runId = 'run_abc123';
14
+ * const marker = generateZendeskMarker(runId);
15
+ * // Returns: '[THID:run_abc123]'
16
+ *
17
+ * // Append to your agent response:
18
+ * const response = `I've found your order. ${marker}`;
19
+ * ```
20
+ */
21
+ export declare function generateZendeskMarker(runId: string): string;
22
+ /**
23
+ * Parse a Zendesk marker from text
24
+ *
25
+ * @example
26
+ * ```typescript
27
+ * const text = 'Thank you for contacting us. [THID:run_abc123]';
28
+ * const runId = parseZendeskMarker(text);
29
+ * // Returns: 'run_abc123'
30
+ * ```
31
+ */
32
+ export declare function parseZendeskMarker(text: string): string | null;
33
+ /**
34
+ * Check if text contains a Zendesk marker
35
+ */
36
+ export declare function hasZendeskMarker(text: string): boolean;
37
+ /**
38
+ * Remove Zendesk marker from text (for clean display)
39
+ *
40
+ * @example
41
+ * ```typescript
42
+ * const text = 'Thank you! [THID:run_abc123]';
43
+ * const clean = removeZendeskMarker(text);
44
+ * // Returns: 'Thank you!'
45
+ * ```
46
+ */
47
+ export declare function removeZendeskMarker(text: string): string;
48
+ /**
49
+ * Confidence scores for each link method
50
+ * Based on deterministic linking principles from v3 spec
51
+ */
52
+ export declare const LINK_METHOD_CONFIDENCE: Record<LinkMethod, number>;
53
+ /**
54
+ * Link evidence structure
55
+ */
56
+ export interface LinkEvidence {
57
+ method: LinkMethod;
58
+ confidence: number;
59
+ timestamp: string;
60
+ details: {
61
+ runId?: string;
62
+ ticketId?: string;
63
+ externalTicketId?: string;
64
+ platform?: string;
65
+ sessionId?: string;
66
+ email?: string;
67
+ timeWindowMinutes?: number;
68
+ customFieldName?: string;
69
+ customFieldValue?: string;
70
+ };
71
+ }
72
+ /**
73
+ * Create link input
74
+ */
75
+ export interface CreateLinkInput {
76
+ runId: string;
77
+ ticketId?: string;
78
+ externalTicketId?: string;
79
+ platform?: 'zendesk' | 'intercom' | 'salesforce' | 'freshdesk';
80
+ method: LinkMethod;
81
+ evidence?: Partial<LinkEvidence['details']>;
82
+ }
83
+ /**
84
+ * Link response
85
+ */
86
+ export interface LinkResponse {
87
+ id: string;
88
+ runId: string;
89
+ ticketId?: string;
90
+ externalTicketId?: string;
91
+ platform?: string;
92
+ method: LinkMethod;
93
+ confidence: number;
94
+ evidence: LinkEvidence;
95
+ createdAt: string;
96
+ }
97
+ /**
98
+ * Linking API client
99
+ */
100
+ export declare const linking: {
101
+ /**
102
+ * Create a link between a run and a ticket
103
+ *
104
+ * @example
105
+ * ```typescript
106
+ * // SDK explicit linking (highest confidence)
107
+ * const link = await linking.create({
108
+ * runId: 'run_abc123',
109
+ * ticketId: 'ticket_xyz',
110
+ * method: 'sdk_explicit',
111
+ * });
112
+ *
113
+ * // Zendesk marker linking
114
+ * const link = await linking.create({
115
+ * runId: 'run_abc123',
116
+ * externalTicketId: '12345',
117
+ * platform: 'zendesk',
118
+ * method: 'zendesk_marker',
119
+ * });
120
+ * ```
121
+ */
122
+ create(input: CreateLinkInput): Promise<LinkResponse>;
123
+ /**
124
+ * Get links for a run
125
+ *
126
+ * @example
127
+ * ```typescript
128
+ * const links = await linking.getForRun('run_abc123');
129
+ * ```
130
+ */
131
+ getForRun(runId: string): Promise<LinkResponse[]>;
132
+ /**
133
+ * Get links for a ticket
134
+ *
135
+ * @example
136
+ * ```typescript
137
+ * const links = await linking.getForTicket('ticket_xyz');
138
+ * ```
139
+ */
140
+ getForTicket(ticketId: string): Promise<LinkResponse[]>;
141
+ /**
142
+ * Verify a link (confirm or reject)
143
+ *
144
+ * @example
145
+ * ```typescript
146
+ * await linking.verify('link_abc123', {
147
+ * verified: true,
148
+ * notes: 'Confirmed by support agent',
149
+ * });
150
+ * ```
151
+ */
152
+ verify(linkId: string, options: {
153
+ verified: boolean;
154
+ notes?: string;
155
+ }): Promise<{
156
+ linkId: string;
157
+ verified: boolean;
158
+ message: string;
159
+ }>;
160
+ /**
161
+ * Delete a link
162
+ *
163
+ * @example
164
+ * ```typescript
165
+ * await linking.delete('link_abc123');
166
+ * ```
167
+ */
168
+ delete(linkId: string): Promise<void>;
169
+ /**
170
+ * Auto-link runs to tickets based on available evidence
171
+ *
172
+ * @example
173
+ * ```typescript
174
+ * const results = await linking.autoLink('run_abc123');
175
+ * for (const link of results.created) {
176
+ * console.log(`Linked to ${link.ticketId} via ${link.method}`);
177
+ * }
178
+ * ```
179
+ */
180
+ autoLink(runId: string): Promise<{
181
+ runId: string;
182
+ created: LinkResponse[];
183
+ candidates: Array<{
184
+ ticketId: string;
185
+ method: LinkMethod;
186
+ confidence: number;
187
+ reason: string;
188
+ }>;
189
+ }>;
190
+ };
191
+ /**
192
+ * Create SDK explicit link (convenience function)
193
+ */
194
+ export declare function linkRunToTicket(runId: string, ticketId: string): Promise<LinkResponse>;
195
+ /**
196
+ * Create Zendesk link via marker
197
+ */
198
+ export declare function linkRunToZendeskTicket(runId: string, zendeskTicketId: string): Promise<LinkResponse>;
199
+ /**
200
+ * Get the best link method for a given scenario
201
+ */
202
+ export declare function getBestLinkMethod(available: {
203
+ hasTicketId: boolean;
204
+ hasMarker: boolean;
205
+ hasCustomField: boolean;
206
+ hasMiddlewareStamp: boolean;
207
+ hasSessionId: boolean;
208
+ hasEmail: boolean;
209
+ }): LinkMethod | null;
210
+ /**
211
+ * Format link confidence for display
212
+ */
213
+ export declare function formatLinkConfidence(confidence: number): string;
214
+ /**
215
+ * Get confidence level label
216
+ */
217
+ export declare function getConfidenceLevel(confidence: number): 'definitive' | 'high' | 'medium' | 'low';