@thinkhive/sdk 4.0.1 → 4.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +364 -208
- package/dist/api/claims.js +2 -3
- package/dist/api/documents.d.ts +52 -0
- package/dist/api/documents.js +58 -0
- package/dist/api/drift.d.ts +70 -0
- package/dist/api/drift.js +71 -0
- package/dist/api/eval-runs.d.ts +126 -0
- package/dist/api/eval-runs.js +126 -0
- package/dist/api/llm-costs.d.ts +104 -0
- package/dist/api/llm-costs.js +81 -0
- package/dist/api/notifications.d.ts +103 -0
- package/dist/api/notifications.js +110 -0
- package/dist/api/roi-analytics.d.ts +99 -0
- package/dist/api/roi-analytics.js +64 -1
- package/dist/api/sessions.d.ts +57 -0
- package/dist/api/sessions.js +49 -0
- package/dist/api/shadow-tests.d.ts +78 -0
- package/dist/api/shadow-tests.js +80 -0
- package/dist/api/signals.d.ts +177 -0
- package/dist/api/signals.js +172 -0
- package/dist/core/client.d.ts +2 -0
- package/dist/core/client.js +10 -4
- package/dist/core/config.d.ts +1 -1
- package/dist/core/config.js +2 -2
- package/dist/guardrails.d.ts +70 -0
- package/dist/guardrails.js +34 -0
- package/dist/index.d.ts +209 -18
- package/dist/index.js +55 -7
- package/dist/integrations/ticket-linking.d.ts +23 -17
- package/dist/integrations/ticket-linking.js +95 -12
- package/package.json +1 -1
|
@@ -95,6 +95,49 @@ exports.LINK_METHOD_CONFIDENCE = {
|
|
|
95
95
|
email_time_window: 0.6, // Email + 15min window (lowest confidence)
|
|
96
96
|
manual: 1.0, // Human-assigned
|
|
97
97
|
};
|
|
98
|
+
/**
|
|
99
|
+
* Transform V3 evidence ({ type, data }) into SDK LinkEvidence shape ({ method, confidence, timestamp, details })
|
|
100
|
+
*/
|
|
101
|
+
function normalizeEvidence(raw, method, confidence, createdAt) {
|
|
102
|
+
// V3 API returns evidence as { type: string, data: Record<string, unknown> }
|
|
103
|
+
const v3Evidence = raw;
|
|
104
|
+
const data = v3Evidence?.data ?? {};
|
|
105
|
+
return {
|
|
106
|
+
method,
|
|
107
|
+
confidence,
|
|
108
|
+
timestamp: data.callTimestamp ?? data.verifiedAt ?? createdAt,
|
|
109
|
+
details: {
|
|
110
|
+
runId: data.runId,
|
|
111
|
+
ticketId: data.ticketId,
|
|
112
|
+
externalTicketId: data.externalTicketId,
|
|
113
|
+
platform: data.platform,
|
|
114
|
+
sessionId: data.sessionId,
|
|
115
|
+
email: (data.customerEmail ?? data.email),
|
|
116
|
+
timeWindowMinutes: data.timeWindowSeconds != null ? Number(data.timeWindowSeconds) / 60 : undefined,
|
|
117
|
+
customFieldName: data.fieldName,
|
|
118
|
+
customFieldValue: data.fieldValue,
|
|
119
|
+
},
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Normalize a raw v3 link into the SDK's LinkResponse shape
|
|
124
|
+
*/
|
|
125
|
+
function normalizeLink(raw) {
|
|
126
|
+
const method = raw.linkMethod;
|
|
127
|
+
const parsedConfidence = typeof raw.confidence === 'string' ? Number(raw.confidence) : raw.confidence;
|
|
128
|
+
const confidence = Number.isFinite(parsedConfidence) ? parsedConfidence : (exports.LINK_METHOD_CONFIDENCE[method] ?? 0);
|
|
129
|
+
return {
|
|
130
|
+
id: raw.id,
|
|
131
|
+
runId: (raw.runId ?? raw.traceId ?? ''),
|
|
132
|
+
ticketId: raw.ticketId ?? undefined,
|
|
133
|
+
externalTicketId: raw.ticketExternalId ?? undefined,
|
|
134
|
+
platform: raw.ticketExternalSource ?? undefined,
|
|
135
|
+
method,
|
|
136
|
+
confidence,
|
|
137
|
+
evidence: normalizeEvidence(raw.evidence, method, confidence, raw.createdAt),
|
|
138
|
+
createdAt: raw.createdAt,
|
|
139
|
+
};
|
|
140
|
+
}
|
|
98
141
|
/**
|
|
99
142
|
* Linking API client
|
|
100
143
|
*/
|
|
@@ -121,13 +164,34 @@ exports.linking = {
|
|
|
121
164
|
* ```
|
|
122
165
|
*/
|
|
123
166
|
async create(input) {
|
|
124
|
-
|
|
167
|
+
// Build method-specific evidence data
|
|
168
|
+
const baseEvidence = {
|
|
169
|
+
...input.evidence,
|
|
170
|
+
sdkVersion: '4.0.0',
|
|
171
|
+
callTimestamp: new Date().toISOString(),
|
|
172
|
+
};
|
|
173
|
+
// For sdk_explicit, add ticketIdSource (required by SdkExplicitEvidenceSchema)
|
|
174
|
+
const evidenceData = input.method === 'sdk_explicit'
|
|
175
|
+
? { ...baseEvidence, ticketIdSource: 'sdk_parameter' }
|
|
176
|
+
: baseEvidence;
|
|
177
|
+
const raw = await (0, client_1.apiRequestWithData)('/links', {
|
|
125
178
|
method: 'POST',
|
|
179
|
+
apiVersion: 'v3',
|
|
126
180
|
body: {
|
|
127
|
-
|
|
181
|
+
traceId: input.runId, // V3 maps run to trace
|
|
182
|
+
runId: input.runId,
|
|
183
|
+
ticketId: input.ticketId,
|
|
184
|
+
ticketExternalId: input.externalTicketId,
|
|
185
|
+
ticketExternalSource: input.platform,
|
|
186
|
+
linkMethod: input.method,
|
|
128
187
|
confidence: exports.LINK_METHOD_CONFIDENCE[input.method],
|
|
188
|
+
evidence: {
|
|
189
|
+
type: input.method,
|
|
190
|
+
data: evidenceData,
|
|
191
|
+
},
|
|
129
192
|
},
|
|
130
193
|
});
|
|
194
|
+
return normalizeLink(raw);
|
|
131
195
|
},
|
|
132
196
|
/**
|
|
133
197
|
* Get links for a run
|
|
@@ -138,7 +202,9 @@ exports.linking = {
|
|
|
138
202
|
* ```
|
|
139
203
|
*/
|
|
140
204
|
async getForRun(runId) {
|
|
141
|
-
|
|
205
|
+
const params = new URLSearchParams({ runId });
|
|
206
|
+
const result = await (0, client_1.apiRequestWithData)(`/links?${params}`, { apiVersion: 'v3' });
|
|
207
|
+
return (result.links || []).map(normalizeLink);
|
|
142
208
|
},
|
|
143
209
|
/**
|
|
144
210
|
* Get links for a ticket
|
|
@@ -149,7 +215,9 @@ exports.linking = {
|
|
|
149
215
|
* ```
|
|
150
216
|
*/
|
|
151
217
|
async getForTicket(ticketId) {
|
|
152
|
-
|
|
218
|
+
const params = new URLSearchParams({ ticketId });
|
|
219
|
+
const result = await (0, client_1.apiRequestWithData)(`/links?${params}`, { apiVersion: 'v3' });
|
|
220
|
+
return (result.links || []).map(normalizeLink);
|
|
153
221
|
},
|
|
154
222
|
/**
|
|
155
223
|
* Verify a link (confirm or reject)
|
|
@@ -163,10 +231,12 @@ exports.linking = {
|
|
|
163
231
|
* ```
|
|
164
232
|
*/
|
|
165
233
|
async verify(linkId, options) {
|
|
166
|
-
|
|
234
|
+
const raw = await (0, client_1.apiRequestWithData)(`/links/${linkId}/verify`, {
|
|
167
235
|
method: 'POST',
|
|
168
|
-
|
|
236
|
+
apiVersion: 'v3',
|
|
237
|
+
body: { linkId, ...options },
|
|
169
238
|
});
|
|
239
|
+
return normalizeLink(raw);
|
|
170
240
|
},
|
|
171
241
|
/**
|
|
172
242
|
* Delete a link
|
|
@@ -177,22 +247,35 @@ exports.linking = {
|
|
|
177
247
|
* ```
|
|
178
248
|
*/
|
|
179
249
|
async delete(linkId) {
|
|
180
|
-
await (0, client_1.apiRequest)(`/links/${linkId}`, { method: 'DELETE' });
|
|
250
|
+
await (0, client_1.apiRequest)(`/links/${linkId}`, { method: 'DELETE', apiVersion: 'v3' });
|
|
181
251
|
},
|
|
182
252
|
/**
|
|
183
|
-
* Auto-link
|
|
253
|
+
* Auto-link a run to a ticket based on available evidence
|
|
184
254
|
*
|
|
185
255
|
* @example
|
|
186
256
|
* ```typescript
|
|
187
|
-
* const
|
|
188
|
-
*
|
|
257
|
+
* const link = await linking.autoLink('run_abc123');
|
|
258
|
+
* if (link) {
|
|
189
259
|
* console.log(`Linked to ${link.ticketId} via ${link.method}`);
|
|
190
260
|
* }
|
|
191
261
|
* ```
|
|
192
262
|
*/
|
|
193
263
|
async autoLink(runId) {
|
|
194
|
-
|
|
264
|
+
const raw = await (0, client_1.apiRequestWithData)(`/links/auto`, {
|
|
265
|
+
method: 'POST',
|
|
266
|
+
apiVersion: 'v3',
|
|
267
|
+
body: { traceId: runId, hints: {} },
|
|
268
|
+
});
|
|
269
|
+
return raw ? normalizeLink(raw) : null;
|
|
270
|
+
},
|
|
271
|
+
async stats() {
|
|
272
|
+
return (0, client_1.apiRequestWithData)('/links/stats', { apiVersion: 'v3' });
|
|
273
|
+
},
|
|
274
|
+
async generateMarker(options) {
|
|
275
|
+
return (0, client_1.apiRequestWithData)('/links/zendesk-marker', {
|
|
195
276
|
method: 'POST',
|
|
277
|
+
apiVersion: 'v3',
|
|
278
|
+
body: options,
|
|
196
279
|
});
|
|
197
280
|
},
|
|
198
281
|
};
|
|
@@ -256,4 +339,4 @@ function getConfidenceLevel(confidence) {
|
|
|
256
339
|
return 'medium';
|
|
257
340
|
return 'low';
|
|
258
341
|
}
|
|
259
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ticket-linking.js","sourceRoot":"","sources":["../../src/integrations/ticket-linking.ts"],"names":[],"mappings":";AAAA;;;;;GAKG;;;AA8BH,sDAKC;AAYD,gDAGC;AAKD,4CAEC;AAYD,kDAEC;AAqMD,0CASC;AAKD,wDAUC;AAKD,8CAiBC;AAKD,oDAEC;AAKD,gDAOC;AA3UD,2CAAgE;AAGhE,+EAA+E;AAC/E,iBAAiB;AACjB,+EAA+E;AAE/E;;;GAGG;AACH,MAAM,aAAa,GAAG,QAAQ,CAAC;AAC/B,MAAM,aAAa,GAAG,GAAG,CAAC;AAC1B,MAAM,YAAY,GAAG,2BAA2B,CAAC;AAEjD;;;;;;;;;;;;GAYG;AACH,SAAgB,qBAAqB,CAAC,KAAa;IACjD,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;IACnE,CAAC;IACD,OAAO,GAAG,aAAa,GAAG,KAAK,GAAG,aAAa,EAAE,CAAC;AACpD,CAAC;AAED;;;;;;;;;GASG;AACH,SAAgB,kBAAkB,CAAC,IAAY;IAC7C,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;IACvC,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;AACjC,CAAC;AAED;;GAEG;AACH,SAAgB,gBAAgB,CAAC,IAAY;IAC3C,OAAO,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACjC,CAAC;AAED;;;;;;;;;GASG;AACH,SAAgB,mBAAmB,CAAC,IAAY;IAC9C,OAAO,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;AAC/C,CAAC;AAED,+EAA+E;AAC/E,4BAA4B;AAC5B,+EAA+E;AAE/E;;;GAGG;AACU,QAAA,sBAAsB,GAA+B;IAChE,YAAY,EAAE,GAAG,EAAO,iCAAiC;IACzD,cAAc,EAAE,GAAG,EAAK,mCAAmC;IAC3D,YAAY,EAAE,GAAG,EAAO,uBAAuB;IAC/C,gBAAgB,EAAE,IAAI,EAAE,+BAA+B;IACvD,aAAa,EAAE,IAAI,EAAK,yBAAyB;IACjD,iBAAiB,EAAE,GAAG,EAAE,2CAA2C;IACnE,MAAM,EAAE,GAAG,EAAa,iBAAiB;CAC1C,CAAC;AAqDF;;GAEG;AACU,QAAA,OAAO,GAAG;IACrB;;;;;;;;;;;;;;;;;;;;OAoBG;IACH,KAAK,CAAC,MAAM,CAAC,KAAsB;QACjC,OAAO,IAAA,2BAAkB,EAAe,QAAQ,EAAE;YAChD,MAAM,EAAE,MAAM;YACd,IAAI,EAAE;gBACJ,GAAG,KAAK;gBACR,UAAU,EAAE,8BAAsB,CAAC,KAAK,CAAC,MAAM,CAAC;aACjD;SACF,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,SAAS,CAAC,KAAa;QAC3B,OAAO,IAAA,2BAAkB,EAAiB,SAAS,KAAK,QAAQ,CAAC,CAAC;IACpE,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,YAAY,CAAC,QAAgB;QACjC,OAAO,IAAA,2BAAkB,EAAiB,YAAY,QAAQ,QAAQ,CAAC,CAAC;IAC1E,CAAC;IAED;;;;;;;;;;OAUG;IACH,KAAK,CAAC,MAAM,CACV,MAAc,EACd,OAA8C;QAE9C,OAAO,IAAA,2BAAkB,EAAC,UAAU,MAAM,SAAS,EAAE;YACnD,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,OAAO;SACd,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,MAAM,CAAC,MAAc;QACzB,MAAM,IAAA,mBAAU,EAAC,UAAU,MAAM,EAAE,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,CAAC;IAC7D,CAAC;IAED;;;;;;;;;;OAUG;IACH,KAAK,CAAC,QAAQ,CAAC,KAAa;QAU1B,OAAO,IAAA,2BAAkB,EAAC,SAAS,KAAK,YAAY,EAAE;YACpD,MAAM,EAAE,MAAM;SACf,CAAC,CAAC;IACL,CAAC;CACF,CAAC;AAEF,+EAA+E;AAC/E,mBAAmB;AACnB,+EAA+E;AAE/E;;GAEG;AACI,KAAK,UAAU,eAAe,CACnC,KAAa,EACb,QAAgB;IAEhB,OAAO,eAAO,CAAC,MAAM,CAAC;QACpB,KAAK;QACL,QAAQ;QACR,MAAM,EAAE,cAAc;KACvB,CAAC,CAAC;AACL,CAAC;AAED;;GAEG;AACI,KAAK,UAAU,sBAAsB,CAC1C,KAAa,EACb,eAAuB;IAEvB,OAAO,eAAO,CAAC,MAAM,CAAC;QACpB,KAAK;QACL,gBAAgB,EAAE,eAAe;QACjC,QAAQ,EAAE,SAAS;QACnB,MAAM,EAAE,gBAAgB;KACzB,CAAC,CAAC;AACL,CAAC;AAED;;GAEG;AACH,SAAgB,iBAAiB,CAC/B,SAOC;IAED,IAAI,SAAS,CAAC,WAAW;QAAE,OAAO,cAAc,CAAC;IACjD,IAAI,SAAS,CAAC,SAAS;QAAE,OAAO,gBAAgB,CAAC;IACjD,IAAI,SAAS,CAAC,cAAc;QAAE,OAAO,cAAc,CAAC;IACpD,IAAI,SAAS,CAAC,kBAAkB;QAAE,OAAO,kBAAkB,CAAC;IAC5D,IAAI,SAAS,CAAC,YAAY;QAAE,OAAO,eAAe,CAAC;IACnD,IAAI,SAAS,CAAC,QAAQ;QAAE,OAAO,mBAAmB,CAAC;IACnD,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;GAEG;AACH,SAAgB,oBAAoB,CAAC,UAAkB;IACrD,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,GAAG,CAAC,GAAG,CAAC;AAC5C,CAAC;AAED;;GAEG;AACH,SAAgB,kBAAkB,CAChC,UAAkB;IAElB,IAAI,UAAU,IAAI,GAAG;QAAE,OAAO,YAAY,CAAC;IAC3C,IAAI,UAAU,IAAI,IAAI;QAAE,OAAO,MAAM,CAAC;IACtC,IAAI,UAAU,IAAI,GAAG;QAAE,OAAO,QAAQ,CAAC;IACvC,OAAO,KAAK,CAAC;AACf,CAAC","sourcesContent":["/**\n * ThinkHive SDK v3.0 - Ticket Linking\n *\n * Deterministic linking between runs and support tickets\n * 7 link methods with explicit confidence scores\n */\n\nimport { apiRequest, apiRequestWithData } from '../core/client';\nimport type { LinkMethod } from '../core/types';\n\n// ============================================================================\n// ZENDESK MARKER\n// ============================================================================\n\n/**\n * Zendesk marker format: [THID:run_xxx]\n * Embedded in agent responses for deterministic linking\n */\nconst MARKER_PREFIX = '[THID:';\nconst MARKER_SUFFIX = ']';\nconst MARKER_REGEX = /\\[THID:([a-zA-Z0-9_-]+)\\]/;\n\n/**\n * Generate a Zendesk marker to embed in agent responses\n *\n * @example\n * ```typescript\n * const runId = 'run_abc123';\n * const marker = generateZendeskMarker(runId);\n * // Returns: '[THID:run_abc123]'\n *\n * // Append to your agent response:\n * const response = `I've found your order. ${marker}`;\n * ```\n */\nexport function generateZendeskMarker(runId: string): string {\n  if (!runId) {\n    throw new Error('Run ID is required to generate Zendesk marker');\n  }\n  return `${MARKER_PREFIX}${runId}${MARKER_SUFFIX}`;\n}\n\n/**\n * Parse a Zendesk marker from text\n *\n * @example\n * ```typescript\n * const text = 'Thank you for contacting us. [THID:run_abc123]';\n * const runId = parseZendeskMarker(text);\n * // Returns: 'run_abc123'\n * ```\n */\nexport function parseZendeskMarker(text: string): string | null {\n  const match = text.match(MARKER_REGEX);\n  return match ? match[1] : null;\n}\n\n/**\n * Check if text contains a Zendesk marker\n */\nexport function hasZendeskMarker(text: string): boolean {\n  return MARKER_REGEX.test(text);\n}\n\n/**\n * Remove Zendesk marker from text (for clean display)\n *\n * @example\n * ```typescript\n * const text = 'Thank you! [THID:run_abc123]';\n * const clean = removeZendeskMarker(text);\n * // Returns: 'Thank you!'\n * ```\n */\nexport function removeZendeskMarker(text: string): string {\n  return text.replace(MARKER_REGEX, '').trim();\n}\n\n// ============================================================================\n// LINK METHODS & CONFIDENCE\n// ============================================================================\n\n/**\n * Confidence scores for each link method\n * Based on deterministic linking principles from v3 spec\n */\nexport const LINK_METHOD_CONFIDENCE: Record<LinkMethod, number> = {\n  sdk_explicit: 1.0,      // Direct SDK call with ticket ID\n  zendesk_marker: 1.0,    // Embedded THID marker in response\n  custom_field: 1.0,      // Zendesk custom field\n  middleware_stamp: 0.98, // Middleware-injected trace ID\n  session_match: 0.95,    // Session ID correlation\n  email_time_window: 0.6, // Email + 15min window (lowest confidence)\n  manual: 1.0,            // Human-assigned\n};\n\n/**\n * Link evidence structure\n */\nexport interface LinkEvidence {\n  method: LinkMethod;\n  confidence: number;\n  timestamp: string;\n  details: {\n    runId?: string;\n    ticketId?: string;\n    externalTicketId?: string;\n    platform?: string;\n    sessionId?: string;\n    email?: string;\n    timeWindowMinutes?: number;\n    customFieldName?: string;\n    customFieldValue?: string;\n  };\n}\n\n// ============================================================================\n// LINKING API\n// ============================================================================\n\n/**\n * Create link input\n */\nexport interface CreateLinkInput {\n  runId: string;\n  ticketId?: string;\n  externalTicketId?: string;\n  platform?: 'zendesk' | 'intercom' | 'salesforce' | 'freshdesk';\n  method: LinkMethod;\n  evidence?: Partial<LinkEvidence['details']>;\n}\n\n/**\n * Link response\n */\nexport interface LinkResponse {\n  id: string;\n  runId: string;\n  ticketId?: string;\n  externalTicketId?: string;\n  platform?: string;\n  method: LinkMethod;\n  confidence: number;\n  evidence: LinkEvidence;\n  createdAt: string;\n}\n\n/**\n * Linking API client\n */\nexport const linking = {\n  /**\n   * Create a link between a run and a ticket\n   *\n   * @example\n   * ```typescript\n   * // SDK explicit linking (highest confidence)\n   * const link = await linking.create({\n   *   runId: 'run_abc123',\n   *   ticketId: 'ticket_xyz',\n   *   method: 'sdk_explicit',\n   * });\n   *\n   * // Zendesk marker linking\n   * const link = await linking.create({\n   *   runId: 'run_abc123',\n   *   externalTicketId: '12345',\n   *   platform: 'zendesk',\n   *   method: 'zendesk_marker',\n   * });\n   * ```\n   */\n  async create(input: CreateLinkInput): Promise<LinkResponse> {\n    return apiRequestWithData<LinkResponse>('/links', {\n      method: 'POST',\n      body: {\n        ...input,\n        confidence: LINK_METHOD_CONFIDENCE[input.method],\n      },\n    });\n  },\n\n  /**\n   * Get links for a run\n   *\n   * @example\n   * ```typescript\n   * const links = await linking.getForRun('run_abc123');\n   * ```\n   */\n  async getForRun(runId: string): Promise<LinkResponse[]> {\n    return apiRequestWithData<LinkResponse[]>(`/runs/${runId}/links`);\n  },\n\n  /**\n   * Get links for a ticket\n   *\n   * @example\n   * ```typescript\n   * const links = await linking.getForTicket('ticket_xyz');\n   * ```\n   */\n  async getForTicket(ticketId: string): Promise<LinkResponse[]> {\n    return apiRequestWithData<LinkResponse[]>(`/tickets/${ticketId}/links`);\n  },\n\n  /**\n   * Verify a link (confirm or reject)\n   *\n   * @example\n   * ```typescript\n   * await linking.verify('link_abc123', {\n   *   verified: true,\n   *   notes: 'Confirmed by support agent',\n   * });\n   * ```\n   */\n  async verify(\n    linkId: string,\n    options: { verified: boolean; notes?: string }\n  ): Promise<{ linkId: string; verified: boolean; message: string }> {\n    return apiRequestWithData(`/links/${linkId}/verify`, {\n      method: 'POST',\n      body: options,\n    });\n  },\n\n  /**\n   * Delete a link\n   *\n   * @example\n   * ```typescript\n   * await linking.delete('link_abc123');\n   * ```\n   */\n  async delete(linkId: string): Promise<void> {\n    await apiRequest(`/links/${linkId}`, { method: 'DELETE' });\n  },\n\n  /**\n   * Auto-link runs to tickets based on available evidence\n   *\n   * @example\n   * ```typescript\n   * const results = await linking.autoLink('run_abc123');\n   * for (const link of results.created) {\n   *   console.log(`Linked to ${link.ticketId} via ${link.method}`);\n   * }\n   * ```\n   */\n  async autoLink(runId: string): Promise<{\n    runId: string;\n    created: LinkResponse[];\n    candidates: Array<{\n      ticketId: string;\n      method: LinkMethod;\n      confidence: number;\n      reason: string;\n    }>;\n  }> {\n    return apiRequestWithData(`/runs/${runId}/auto-link`, {\n      method: 'POST',\n    });\n  },\n};\n\n// ============================================================================\n// HELPER FUNCTIONS\n// ============================================================================\n\n/**\n * Create SDK explicit link (convenience function)\n */\nexport async function linkRunToTicket(\n  runId: string,\n  ticketId: string\n): Promise<LinkResponse> {\n  return linking.create({\n    runId,\n    ticketId,\n    method: 'sdk_explicit',\n  });\n}\n\n/**\n * Create Zendesk link via marker\n */\nexport async function linkRunToZendeskTicket(\n  runId: string,\n  zendeskTicketId: string\n): Promise<LinkResponse> {\n  return linking.create({\n    runId,\n    externalTicketId: zendeskTicketId,\n    platform: 'zendesk',\n    method: 'zendesk_marker',\n  });\n}\n\n/**\n * Get the best link method for a given scenario\n */\nexport function getBestLinkMethod(\n  available: {\n    hasTicketId: boolean;\n    hasMarker: boolean;\n    hasCustomField: boolean;\n    hasMiddlewareStamp: boolean;\n    hasSessionId: boolean;\n    hasEmail: boolean;\n  }\n): LinkMethod | null {\n  if (available.hasTicketId) return 'sdk_explicit';\n  if (available.hasMarker) return 'zendesk_marker';\n  if (available.hasCustomField) return 'custom_field';\n  if (available.hasMiddlewareStamp) return 'middleware_stamp';\n  if (available.hasSessionId) return 'session_match';\n  if (available.hasEmail) return 'email_time_window';\n  return null;\n}\n\n/**\n * Format link confidence for display\n */\nexport function formatLinkConfidence(confidence: number): string {\n  return `${Math.round(confidence * 100)}%`;\n}\n\n/**\n * Get confidence level label\n */\nexport function getConfidenceLevel(\n  confidence: number\n): 'definitive' | 'high' | 'medium' | 'low' {\n  if (confidence >= 1.0) return 'definitive';\n  if (confidence >= 0.95) return 'high';\n  if (confidence >= 0.8) return 'medium';\n  return 'low';\n}\n"]}
|
|
342
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ticket-linking.js","sourceRoot":"","sources":["../../src/integrations/ticket-linking.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ThinkHive SDK v3.0 - Ticket Linking\n *\n * Deterministic linking between runs and support tickets\n * 7 link methods with explicit confidence scores\n */\n\nimport { apiRequest, apiRequestWithData } from '../core/client';\nimport type { LinkMethod } from '../core/types';\n\n// ============================================================================\n// ZENDESK MARKER\n// ============================================================================\n\n/**\n * Zendesk marker format: [THID:run_xxx]\n * Embedded in agent responses for deterministic linking\n */\nconst MARKER_PREFIX = '[THID:';\nconst MARKER_SUFFIX = ']';\nconst MARKER_REGEX = /\\[THID:([a-zA-Z0-9_-]+)\\]/;\n\n/**\n * Generate a Zendesk marker to embed in agent responses\n *\n * @example\n * ```typescript\n * const runId = 'run_abc123';\n * const marker = generateZendeskMarker(runId);\n * // Returns: '[THID:run_abc123]'\n *\n * // Append to your agent response:\n * const response = `I've found your order. ${marker}`;\n * ```\n */\nexport function generateZendeskMarker(runId: string): string {\n  if (!runId) {\n    throw new Error('Run ID is required to generate Zendesk marker');\n  }\n  return `${MARKER_PREFIX}${runId}${MARKER_SUFFIX}`;\n}\n\n/**\n * Parse a Zendesk marker from text\n *\n * @example\n * ```typescript\n * const text = 'Thank you for contacting us. [THID:run_abc123]';\n * const runId = parseZendeskMarker(text);\n * // Returns: 'run_abc123'\n * ```\n */\nexport function parseZendeskMarker(text: string): string | null {\n  const match = text.match(MARKER_REGEX);\n  return match ? match[1] : null;\n}\n\n/**\n * Check if text contains a Zendesk marker\n */\nexport function hasZendeskMarker(text: string): boolean {\n  return MARKER_REGEX.test(text);\n}\n\n/**\n * Remove Zendesk marker from text (for clean display)\n *\n * @example\n * ```typescript\n * const text = 'Thank you! [THID:run_abc123]';\n * const clean = removeZendeskMarker(text);\n * // Returns: 'Thank you!'\n * ```\n */\nexport function removeZendeskMarker(text: string): string {\n  return text.replace(MARKER_REGEX, '').trim();\n}\n\n// ============================================================================\n// LINK METHODS & CONFIDENCE\n// ============================================================================\n\n/**\n * Confidence scores for each link method\n * Based on deterministic linking principles from v3 spec\n */\nexport const LINK_METHOD_CONFIDENCE: Record<LinkMethod, number> = {\n  sdk_explicit: 1.0,      // Direct SDK call with ticket ID\n  zendesk_marker: 1.0,    // Embedded THID marker in response\n  custom_field: 1.0,      // Zendesk custom field\n  middleware_stamp: 0.98, // Middleware-injected trace ID\n  session_match: 0.95,    // Session ID correlation\n  email_time_window: 0.6, // Email + 15min window (lowest confidence)\n  manual: 1.0,            // Human-assigned\n};\n\n/**\n * Link evidence structure\n */\nexport interface LinkEvidence {\n  method: LinkMethod;\n  confidence: number;\n  timestamp: string;\n  details: {\n    runId?: string;\n    ticketId?: string;\n    externalTicketId?: string;\n    platform?: string;\n    sessionId?: string;\n    email?: string;\n    timeWindowMinutes?: number;\n    customFieldName?: string;\n    customFieldValue?: string;\n  };\n}\n\n// ============================================================================\n// LINKING API\n// ============================================================================\n\n/**\n * Create link input\n */\nexport interface CreateLinkInput {\n  runId: string;\n  ticketId?: string;\n  externalTicketId?: string;\n  platform?: 'zendesk' | 'intercom' | 'salesforce' | 'freshdesk';\n  method: LinkMethod;\n  evidence?: Partial<LinkEvidence['details']>;\n}\n\n/**\n * Link response\n */\nexport interface LinkResponse {\n  id: string;\n  runId: string;\n  ticketId?: string;\n  externalTicketId?: string;\n  platform?: string;\n  method: LinkMethod;\n  confidence: number;\n  evidence: LinkEvidence;\n  createdAt: string;\n}\n\n/**\n * Raw link shape returned by v3 API (differs from SDK's LinkResponse)\n */\ninterface RawV3Link {\n  id: string;\n  traceId?: string;\n  runId?: string | null;\n  ticketId?: string | null;\n  ticketExternalId?: string | null;\n  ticketExternalSource?: string | null;\n  linkMethod: string;\n  confidence: string | number;\n  evidence: unknown;\n  createdAt: string;\n}\n\n/**\n * Transform V3 evidence ({ type, data }) into SDK LinkEvidence shape ({ method, confidence, timestamp, details })\n */\nfunction normalizeEvidence(raw: unknown, method: LinkMethod, confidence: number, createdAt: string): LinkEvidence {\n  // V3 API returns evidence as { type: string, data: Record<string, unknown> }\n  const v3Evidence = raw as { type?: string; data?: Record<string, unknown> } | null;\n  const data = v3Evidence?.data ?? {};\n\n  return {\n    method,\n    confidence,\n    timestamp: (data.callTimestamp as string) ?? (data.verifiedAt as string) ?? createdAt,\n    details: {\n      runId: data.runId as string | undefined,\n      ticketId: data.ticketId as string | undefined,\n      externalTicketId: data.externalTicketId as string | undefined,\n      platform: data.platform as string | undefined,\n      sessionId: data.sessionId as string | undefined,\n      email: (data.customerEmail ?? data.email) as string | undefined,\n      timeWindowMinutes: data.timeWindowSeconds != null ? Number(data.timeWindowSeconds) / 60 : undefined,\n      customFieldName: data.fieldName as string | undefined,\n      customFieldValue: data.fieldValue as string | undefined,\n    },\n  };\n}\n\n/**\n * Normalize a raw v3 link into the SDK's LinkResponse shape\n */\nfunction normalizeLink(raw: RawV3Link): LinkResponse {\n  const method = raw.linkMethod as LinkMethod;\n  const parsedConfidence = typeof raw.confidence === 'string' ? Number(raw.confidence) : raw.confidence;\n  const confidence = Number.isFinite(parsedConfidence) ? parsedConfidence : (LINK_METHOD_CONFIDENCE[method] ?? 0);\n\n  return {\n    id: raw.id,\n    runId: (raw.runId ?? raw.traceId ?? '') as string,\n    ticketId: raw.ticketId ?? undefined,\n    externalTicketId: raw.ticketExternalId ?? undefined,\n    platform: raw.ticketExternalSource ?? undefined,\n    method,\n    confidence,\n    evidence: normalizeEvidence(raw.evidence, method, confidence, raw.createdAt),\n    createdAt: raw.createdAt,\n  };\n}\n\n/**\n * Linking API client\n */\nexport const linking = {\n  /**\n   * Create a link between a run and a ticket\n   *\n   * @example\n   * ```typescript\n   * // SDK explicit linking (highest confidence)\n   * const link = await linking.create({\n   *   runId: 'run_abc123',\n   *   ticketId: 'ticket_xyz',\n   *   method: 'sdk_explicit',\n   * });\n   *\n   * // Zendesk marker linking\n   * const link = await linking.create({\n   *   runId: 'run_abc123',\n   *   externalTicketId: '12345',\n   *   platform: 'zendesk',\n   *   method: 'zendesk_marker',\n   * });\n   * ```\n   */\n  async create(input: CreateLinkInput): Promise<LinkResponse> {\n    // Build method-specific evidence data\n    const baseEvidence = {\n      ...input.evidence,\n      sdkVersion: '4.0.0',\n      callTimestamp: new Date().toISOString(),\n    };\n\n    // For sdk_explicit, add ticketIdSource (required by SdkExplicitEvidenceSchema)\n    const evidenceData = input.method === 'sdk_explicit'\n      ? { ...baseEvidence, ticketIdSource: 'sdk_parameter' as const }\n      : baseEvidence;\n\n    const raw = await apiRequestWithData<RawV3Link>('/links', {\n      method: 'POST',\n      apiVersion: 'v3',\n      body: {\n        traceId: input.runId, // V3 maps run to trace\n        runId: input.runId,\n        ticketId: input.ticketId,\n        ticketExternalId: input.externalTicketId,\n        ticketExternalSource: input.platform,\n        linkMethod: input.method,\n        confidence: LINK_METHOD_CONFIDENCE[input.method],\n        evidence: {\n          type: input.method,\n          data: evidenceData,\n        },\n      },\n    });\n    return normalizeLink(raw);\n  },\n\n  /**\n   * Get links for a run\n   *\n   * @example\n   * ```typescript\n   * const links = await linking.getForRun('run_abc123');\n   * ```\n   */\n  async getForRun(runId: string): Promise<LinkResponse[]> {\n    const params = new URLSearchParams({ runId });\n    const result = await apiRequestWithData<{ links: RawV3Link[] }>(`/links?${params}`, { apiVersion: 'v3' });\n    return (result.links || []).map(normalizeLink);\n  },\n\n  /**\n   * Get links for a ticket\n   *\n   * @example\n   * ```typescript\n   * const links = await linking.getForTicket('ticket_xyz');\n   * ```\n   */\n  async getForTicket(ticketId: string): Promise<LinkResponse[]> {\n    const params = new URLSearchParams({ ticketId });\n    const result = await apiRequestWithData<{ links: RawV3Link[] }>(`/links?${params}`, { apiVersion: 'v3' });\n    return (result.links || []).map(normalizeLink);\n  },\n\n  /**\n   * Verify a link (confirm or reject)\n   *\n   * @example\n   * ```typescript\n   * await linking.verify('link_abc123', {\n   *   verified: true,\n   *   notes: 'Confirmed by support agent',\n   * });\n   * ```\n   */\n  async verify(\n    linkId: string,\n    options: { verified: boolean; notes?: string }\n  ): Promise<LinkResponse> {\n    const raw = await apiRequestWithData<RawV3Link>(`/links/${linkId}/verify`, {\n      method: 'POST',\n      apiVersion: 'v3',\n      body: { linkId, ...options },\n    });\n    return normalizeLink(raw);\n  },\n\n  /**\n   * Delete a link\n   *\n   * @example\n   * ```typescript\n   * await linking.delete('link_abc123');\n   * ```\n   */\n  async delete(linkId: string): Promise<void> {\n    await apiRequest(`/links/${linkId}`, { method: 'DELETE', apiVersion: 'v3' });\n  },\n\n  /**\n   * Auto-link a run to a ticket based on available evidence\n   *\n   * @example\n   * ```typescript\n   * const link = await linking.autoLink('run_abc123');\n   * if (link) {\n   *   console.log(`Linked to ${link.ticketId} via ${link.method}`);\n   * }\n   * ```\n   */\n  async autoLink(runId: string): Promise<LinkResponse | null> {\n    const raw = await apiRequestWithData<RawV3Link | null>(`/links/auto`, {\n      method: 'POST',\n      apiVersion: 'v3',\n      body: { traceId: runId, hints: {} },\n    });\n    return raw ? normalizeLink(raw) : null;\n  },\n\n  async stats(): Promise<{\n    totalLinks: number;\n    byMethod: Record<string, number>;\n    avgConfidence: number;\n    verifiedCount: number;\n    unverifiedCount: number;\n  }> {\n    return apiRequestWithData('/links/stats', { apiVersion: 'v3' });\n  },\n\n  async generateMarker(options: {\n    traceId: string;\n    runId?: string;\n    format?: 'html_comment' | 'base64' | 'custom';\n    customTemplate?: string;\n  }): Promise<{\n    marker: string;\n    format: string;\n    traceId: string;\n    runId: string | null;\n    instructions: string;\n  }> {\n    return apiRequestWithData('/links/zendesk-marker', {\n      method: 'POST',\n      apiVersion: 'v3',\n      body: options,\n    });\n  },\n};\n\n// ============================================================================\n// HELPER FUNCTIONS\n// ============================================================================\n\n/**\n * Create SDK explicit link (convenience function)\n */\nexport async function linkRunToTicket(\n  runId: string,\n  ticketId: string\n): Promise<LinkResponse> {\n  return linking.create({\n    runId,\n    ticketId,\n    method: 'sdk_explicit',\n  });\n}\n\n/**\n * Create Zendesk link via marker\n */\nexport async function linkRunToZendeskTicket(\n  runId: string,\n  zendeskTicketId: string\n): Promise<LinkResponse> {\n  return linking.create({\n    runId,\n    externalTicketId: zendeskTicketId,\n    platform: 'zendesk',\n    method: 'zendesk_marker',\n  });\n}\n\n/**\n * Get the best link method for a given scenario\n */\nexport function getBestLinkMethod(\n  available: {\n    hasTicketId: boolean;\n    hasMarker: boolean;\n    hasCustomField: boolean;\n    hasMiddlewareStamp: boolean;\n    hasSessionId: boolean;\n    hasEmail: boolean;\n  }\n): LinkMethod | null {\n  if (available.hasTicketId) return 'sdk_explicit';\n  if (available.hasMarker) return 'zendesk_marker';\n  if (available.hasCustomField) return 'custom_field';\n  if (available.hasMiddlewareStamp) return 'middleware_stamp';\n  if (available.hasSessionId) return 'session_match';\n  if (available.hasEmail) return 'email_time_window';\n  return null;\n}\n\n/**\n * Format link confidence for display\n */\nexport function formatLinkConfidence(confidence: number): string {\n  return `${Math.round(confidence * 100)}%`;\n}\n\n/**\n * Get confidence level label\n */\nexport function getConfidenceLevel(\n  confidence: number\n): 'definitive' | 'high' | 'medium' | 'low' {\n  if (confidence >= 1.0) return 'definitive';\n  if (confidence >= 0.95) return 'high';\n  if (confidence >= 0.8) return 'medium';\n  return 'low';\n}\n"]}
|
package/package.json
CHANGED