genassist-chat-react 1.0.26 → 1.0.28
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 +2 -0
- package/dist/components/GenAgentChat.js +40 -31
- package/dist/components/GenAgentConfigPanel.d.ts +1 -0
- package/dist/components/GenAgentConfigPanel.js +2 -1
- package/dist/components/InteractiveContent.js +3 -2
- package/dist/components/MarkdownMessage.d.ts +11 -0
- package/dist/components/MarkdownMessage.js +56 -0
- package/dist/components/WelcomeCard.js +3 -3
- package/dist/hooks/useChat.d.ts +6 -2
- package/dist/hooks/useChat.js +314 -67
- package/dist/services/chatService.d.ts +24 -2
- package/dist/services/chatService.js +146 -48
- package/dist/types/index.d.ts +22 -0
- package/dist/types/index.js +1 -1
- package/package.json +5 -2
|
@@ -49,8 +49,9 @@ import axios from "axios";
|
|
|
49
49
|
import { createWebSocket, createWebSocketDiagnostic, } from "../utils/websocket";
|
|
50
50
|
export var GENASSIST_AGENT_METADATA_UPDATED = "genassist_agent_metadata_updated";
|
|
51
51
|
var ChatService = /** @class */ (function () {
|
|
52
|
-
function ChatService(baseUrl, apiKey, metadata, tenant, language, useWs) {
|
|
52
|
+
function ChatService(baseUrl, apiKey, metadata, tenant, language, useWs, usePoll) {
|
|
53
53
|
if (useWs === void 0) { useWs = true; }
|
|
54
|
+
if (usePoll === void 0) { usePoll = false; }
|
|
54
55
|
this.conversationId = null;
|
|
55
56
|
this.conversationCreateTime = null; // Track conversation start time
|
|
56
57
|
this.guestToken = null; // Guest token for authorization
|
|
@@ -67,12 +68,14 @@ var ChatService = /** @class */ (function () {
|
|
|
67
68
|
this.thinkingConfig = { phrases: [], delayMs: 1000 };
|
|
68
69
|
this.chatInputMetadata = {};
|
|
69
70
|
this.welcomeObjectUrl = null; // to revoke on reset
|
|
71
|
+
this.usePoll = false;
|
|
70
72
|
this.baseUrl = baseUrl.endsWith("/") ? baseUrl.slice(0, -1) : baseUrl;
|
|
71
73
|
this.apiKey = apiKey;
|
|
72
74
|
this.metadata = metadata;
|
|
73
75
|
this.tenant = tenant;
|
|
74
76
|
this.language = language;
|
|
75
77
|
this.useWs = useWs;
|
|
78
|
+
this.usePoll = usePoll;
|
|
76
79
|
// Try to load a saved conversation for this apiKey from localStorage
|
|
77
80
|
this.loadSavedConversation();
|
|
78
81
|
}
|
|
@@ -80,6 +83,32 @@ var ChatService = /** @class */ (function () {
|
|
|
80
83
|
// Pointer to current conversation metadata for this apiKey
|
|
81
84
|
return "".concat(this.storageKeyBase, ":").concat(this.apiKey);
|
|
82
85
|
};
|
|
86
|
+
/**
|
|
87
|
+
* Handle conversation finalized: emit special message, call finalized handler, persist state.
|
|
88
|
+
*/
|
|
89
|
+
ChatService.prototype.handleConversationFinalized = function () {
|
|
90
|
+
if (this.messageHandler) {
|
|
91
|
+
var now = Date.now() / 1000;
|
|
92
|
+
var finalizedMessage = {
|
|
93
|
+
create_time: now,
|
|
94
|
+
start_time: this.conversationCreateTime
|
|
95
|
+
? now - this.conversationCreateTime
|
|
96
|
+
: 0,
|
|
97
|
+
end_time: this.conversationCreateTime
|
|
98
|
+
? now - this.conversationCreateTime + 0.01
|
|
99
|
+
: 0.01,
|
|
100
|
+
speaker: "special",
|
|
101
|
+
text: "Conversation Finalized",
|
|
102
|
+
type: "finalized",
|
|
103
|
+
};
|
|
104
|
+
this.messageHandler(finalizedMessage);
|
|
105
|
+
}
|
|
106
|
+
if (this.finalizedHandler) {
|
|
107
|
+
this.finalizedHandler();
|
|
108
|
+
}
|
|
109
|
+
this.isFinalized = true;
|
|
110
|
+
this.saveConversation();
|
|
111
|
+
};
|
|
83
112
|
/**
|
|
84
113
|
* Persist agent_chat_input_metadata to localStorage and dispatch a global event.
|
|
85
114
|
* Any consumer (e.g. config panel) can listen for GENASSIST_AGENT_METADATA_UPDATED
|
|
@@ -289,6 +318,27 @@ var ChatService = /** @class */ (function () {
|
|
|
289
318
|
error.response.data.message === "Token has expired." ||
|
|
290
319
|
(typeof error.response.data === "string" && error.response.data.includes("Token has expired"))));
|
|
291
320
|
};
|
|
321
|
+
/**
|
|
322
|
+
* Internal handler for takeover events: emits a special message and invokes takeoverHandler.
|
|
323
|
+
* Runs only once per conversation (idempotent).
|
|
324
|
+
*/
|
|
325
|
+
ChatService.prototype.handleTakeover = function (now) {
|
|
326
|
+
var t = now !== null && now !== void 0 ? now : Date.now() / 1000;
|
|
327
|
+
if (this.messageHandler) {
|
|
328
|
+
var takeoverMessage = {
|
|
329
|
+
create_time: t,
|
|
330
|
+
start_time: this.conversationCreateTime ? t - this.conversationCreateTime : 0,
|
|
331
|
+
end_time: this.conversationCreateTime ? t - this.conversationCreateTime + 0.01 : 0.01,
|
|
332
|
+
speaker: "special",
|
|
333
|
+
text: "Supervisor took over",
|
|
334
|
+
type: "takeover",
|
|
335
|
+
};
|
|
336
|
+
this.messageHandler(takeoverMessage);
|
|
337
|
+
}
|
|
338
|
+
if (this.takeoverHandler) {
|
|
339
|
+
this.takeoverHandler();
|
|
340
|
+
}
|
|
341
|
+
};
|
|
292
342
|
/**
|
|
293
343
|
* Reset the current conversation by clearing the ID and websocket
|
|
294
344
|
*/
|
|
@@ -448,6 +498,7 @@ var ChatService = /** @class */ (function () {
|
|
|
448
498
|
ChatService.prototype.sendMessage = function (message, attachments, extraMetadata, reCaptchaToken) {
|
|
449
499
|
return __awaiter(this, void 0, void 0, function () {
|
|
450
500
|
var now, chatMessage, requestBody, mergedMetadata, response, responseData, i, messageData, agentMessage, error_2, errorMessage;
|
|
501
|
+
var _this = this;
|
|
451
502
|
var _a, _b;
|
|
452
503
|
return __generator(this, function (_c) {
|
|
453
504
|
switch (_c.label) {
|
|
@@ -483,13 +534,23 @@ var ChatService = /** @class */ (function () {
|
|
|
483
534
|
}
|
|
484
535
|
return [4 /*yield*/, axios.patch("".concat(this.baseUrl, "/api/conversations/in-progress/update/").concat(this.conversationId), requestBody, {
|
|
485
536
|
headers: this.getHeaders(),
|
|
537
|
+
}).catch(function (e) {
|
|
538
|
+
var _a;
|
|
539
|
+
var err = (_a = e === null || e === void 0 ? void 0 : e.response) === null || _a === void 0 ? void 0 : _a.data;
|
|
540
|
+
// Handle error when conversation is finalized
|
|
541
|
+
if ((err === null || err === void 0 ? void 0 : err.error_key) === 'CONVERSATION_FINALIZED') {
|
|
542
|
+
_this.handleConversationFinalized();
|
|
543
|
+
}
|
|
544
|
+
else {
|
|
545
|
+
throw e;
|
|
546
|
+
}
|
|
486
547
|
})];
|
|
487
548
|
case 2:
|
|
488
549
|
response = _c.sent();
|
|
489
550
|
// If not using WebSocket, try to retrieve the response message from the update conversation response
|
|
490
|
-
if (!this.useWs && this.messageHandler) {
|
|
551
|
+
if (!this.useWs && this.messageHandler && (!this.useWs || !this.usePoll)) { // usePoll is true when using heartbeat polling
|
|
491
552
|
try {
|
|
492
|
-
responseData = response.data;
|
|
553
|
+
responseData = (response === null || response === void 0 ? void 0 : response.data) || {};
|
|
493
554
|
if (responseData.messages && Array.isArray(responseData.messages)) {
|
|
494
555
|
// Look for the latest agent message in the response
|
|
495
556
|
for (i = responseData.messages.length - 1; i >= 0; i--) {
|
|
@@ -648,52 +709,10 @@ var ChatService = /** @class */ (function () {
|
|
|
648
709
|
}
|
|
649
710
|
}
|
|
650
711
|
else if (data.type === "takeover") {
|
|
651
|
-
|
|
652
|
-
// Create special message for the takeover indicator
|
|
653
|
-
if (_this.messageHandler) {
|
|
654
|
-
var now = Date.now() / 1000;
|
|
655
|
-
var takeoverMessage = {
|
|
656
|
-
create_time: now,
|
|
657
|
-
start_time: _this.conversationCreateTime
|
|
658
|
-
? now - _this.conversationCreateTime
|
|
659
|
-
: 0,
|
|
660
|
-
end_time: _this.conversationCreateTime
|
|
661
|
-
? now - _this.conversationCreateTime + 0.01
|
|
662
|
-
: 0.01,
|
|
663
|
-
speaker: "special",
|
|
664
|
-
text: "Supervisor took over",
|
|
665
|
-
};
|
|
666
|
-
_this.messageHandler(takeoverMessage);
|
|
667
|
-
}
|
|
668
|
-
// Call the takeover handler if provided
|
|
669
|
-
if (_this.takeoverHandler) {
|
|
670
|
-
_this.takeoverHandler();
|
|
671
|
-
}
|
|
712
|
+
_this.handleTakeover();
|
|
672
713
|
}
|
|
673
714
|
else if (data.type === "finalize") {
|
|
674
|
-
|
|
675
|
-
// Create special message for the finalized indicator
|
|
676
|
-
if (_this.messageHandler) {
|
|
677
|
-
var now = Date.now() / 1000;
|
|
678
|
-
var finalizedMessage = {
|
|
679
|
-
create_time: now,
|
|
680
|
-
start_time: _this.conversationCreateTime
|
|
681
|
-
? now - _this.conversationCreateTime
|
|
682
|
-
: 0,
|
|
683
|
-
end_time: _this.conversationCreateTime
|
|
684
|
-
? now - _this.conversationCreateTime + 0.01
|
|
685
|
-
: 0.01,
|
|
686
|
-
speaker: "special",
|
|
687
|
-
text: "Conversation Finalized",
|
|
688
|
-
};
|
|
689
|
-
_this.messageHandler(finalizedMessage);
|
|
690
|
-
}
|
|
691
|
-
// Call the finalized handler if provided
|
|
692
|
-
if (_this.finalizedHandler) {
|
|
693
|
-
_this.finalizedHandler();
|
|
694
|
-
}
|
|
695
|
-
_this.isFinalized = true;
|
|
696
|
-
_this.saveConversation();
|
|
715
|
+
_this.handleConversationFinalized();
|
|
697
716
|
}
|
|
698
717
|
}
|
|
699
718
|
catch (error) {
|
|
@@ -804,7 +823,86 @@ var ChatService = /** @class */ (function () {
|
|
|
804
823
|
}
|
|
805
824
|
return __assign(__assign({}, message), { start_time: message.start_time - this.conversationCreateTime, end_time: message.end_time - this.conversationCreateTime });
|
|
806
825
|
};
|
|
826
|
+
/************ Heartbeat polling ************/
|
|
827
|
+
/**
|
|
828
|
+
* Poll in-progress conversation (heartbeat when WebSocket is disabled).
|
|
829
|
+
* Returns status and messages normalized to ChatMessage[].
|
|
830
|
+
*/
|
|
831
|
+
ChatService.prototype.pollInProgressConversation = function () {
|
|
832
|
+
return __awaiter(this, void 0, void 0, function () {
|
|
833
|
+
var response, pollMessages;
|
|
834
|
+
var _this = this;
|
|
835
|
+
return __generator(this, function (_a) {
|
|
836
|
+
switch (_a.label) {
|
|
837
|
+
case 0:
|
|
838
|
+
if (!this.conversationId || !this.conversationCreateTime) {
|
|
839
|
+
throw new Error("Conversation not started");
|
|
840
|
+
}
|
|
841
|
+
return [4 /*yield*/, axios.get("".concat(this.baseUrl, "/api/conversations/in-progress/poll/").concat(this.conversationId), { headers: this.getHeaders() })];
|
|
842
|
+
case 1:
|
|
843
|
+
response = _a.sent();
|
|
844
|
+
pollMessages = (response.data.messages || []).map(function (m) {
|
|
845
|
+
return _this.normalizePollMessageToChatMessage(m);
|
|
846
|
+
});
|
|
847
|
+
return [2 /*return*/, { status: response.data.status, pollMessages: pollMessages }];
|
|
848
|
+
}
|
|
849
|
+
});
|
|
850
|
+
});
|
|
851
|
+
};
|
|
852
|
+
ChatService.prototype.normalizePollMessageToChatMessage = function (m) {
|
|
853
|
+
var createTime = typeof m.create_time === "number"
|
|
854
|
+
? m.create_time
|
|
855
|
+
: m.create_time
|
|
856
|
+
? +new Date(m.create_time)
|
|
857
|
+
: Math.floor(Date.now() / 1000);
|
|
858
|
+
// Normalize to seconds: ISO string gives ms (>1e12); if so convert. Otherwise assume already seconds.
|
|
859
|
+
if (createTime >= 1e12) {
|
|
860
|
+
createTime = Math.round(createTime / 1000);
|
|
861
|
+
}
|
|
862
|
+
var startTime = this.conversationCreateTime != null
|
|
863
|
+
? createTime - this.conversationCreateTime
|
|
864
|
+
: m.start_time;
|
|
865
|
+
var endTime = this.conversationCreateTime != null
|
|
866
|
+
? startTime + (m.end_time - m.start_time) || 0.01
|
|
867
|
+
: m.end_time;
|
|
868
|
+
return {
|
|
869
|
+
id: m.id,
|
|
870
|
+
create_time: createTime,
|
|
871
|
+
start_time: startTime,
|
|
872
|
+
end_time: endTime,
|
|
873
|
+
speaker: m.speaker,
|
|
874
|
+
text: m.text,
|
|
875
|
+
type: m.type,
|
|
876
|
+
feedback: m.feedback,
|
|
877
|
+
};
|
|
878
|
+
};
|
|
879
|
+
ChatService.prototype.notifyTakeoverFromPoll = function () {
|
|
880
|
+
return __awaiter(this, void 0, void 0, function () {
|
|
881
|
+
return __generator(this, function (_a) {
|
|
882
|
+
// Poll already adds the "Supervisor took over" message from the message list.
|
|
883
|
+
// Only run the handler so UI state updates; do not call handleTakeover() or we duplicate the message.
|
|
884
|
+
if (this.takeoverHandler) {
|
|
885
|
+
this.takeoverHandler();
|
|
886
|
+
}
|
|
887
|
+
return [2 /*return*/];
|
|
888
|
+
});
|
|
889
|
+
});
|
|
890
|
+
};
|
|
891
|
+
ChatService.prototype.notifyFinalizedFromPoll = function () {
|
|
892
|
+
return __awaiter(this, void 0, void 0, function () {
|
|
893
|
+
return __generator(this, function (_a) {
|
|
894
|
+
// Poll already adds the "Conversation Finalized" message from the message list.
|
|
895
|
+
// Only update state and run the handler; do not call handleConversationFinalized() or we duplicate the message.
|
|
896
|
+
if (this.finalizedHandler) {
|
|
897
|
+
this.finalizedHandler();
|
|
898
|
+
}
|
|
899
|
+
this.isFinalized = true;
|
|
900
|
+
this.saveConversation();
|
|
901
|
+
return [2 /*return*/];
|
|
902
|
+
});
|
|
903
|
+
});
|
|
904
|
+
};
|
|
807
905
|
return ChatService;
|
|
808
906
|
}());
|
|
809
907
|
export { ChatService };
|
|
810
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chatService.js","sourceRoot":"","sources":["../../src/services/chatService.ts"],"names":[],"mappings":"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","sourcesContent":["import axios from \"axios\";\nimport {\n  ChatMessage,\n  StartConversationResponse,\n  Attachment,\n  AgentThinkingConfig,\n  AgentWelcomeData,\n  FileUploadResponse,\n} from \"../types\";\nimport {\n  createWebSocket,\n  createWebSocketDiagnostic,\n} from \"../utils/websocket\";\nexport const GENASSIST_AGENT_METADATA_UPDATED = \"genassist_agent_metadata_updated\";\n\nexport class ChatService {\n  private baseUrl: string;\n  private apiKey: string;\n  private metadata: Record<string, any> | undefined;\n  private conversationId: string | null = null;\n  private conversationCreateTime: number | null = null; // Track conversation start time\n  private guestToken: string | null = null; // Guest token for authorization\n  private isFinalized: boolean = false;\n  private webSocket: WebSocket | null = null;\n  private messageHandler: ((message: ChatMessage) => void) | null = null;\n  private takeoverHandler: (() => void) | null = null;\n  private finalizedHandler: (() => void) | null = null;\n  private connectionStateHandler:\n    | ((state: \"connecting\" | \"connected\" | \"disconnected\") => void)\n    | null = null;\n  private welcomeDataHandler: ((data: AgentWelcomeData) => void) | null = null;\n  private storageKeyBase = \"genassist_conversation\";\n  private possibleQueries: string[] = [];\n  private welcomeData: AgentWelcomeData = {};\n  private thinkingConfig: AgentThinkingConfig = { phrases: [], delayMs: 1000 };\n  private chatInputMetadata: Record<string, unknown> = {};\n  private welcomeObjectUrl: string | null = null; // to revoke on reset\n  private tenant: string | undefined;\n  private agentId: string | undefined;\n  private language: string | undefined;\n  private useWs: boolean;\n  private serverUnavailableMessage: string | undefined;\n  private serverUnavailableContactUrl: string | undefined;\n  private serverUnavailableContactLabel: string | undefined;\n\n  constructor(\n    baseUrl: string,\n    apiKey: string,\n    metadata?: Record<string, any>,\n    tenant?: string,\n    language?: string,\n    useWs: boolean = true\n  ) {\n    this.baseUrl = baseUrl.endsWith(\"/\") ? baseUrl.slice(0, -1) : baseUrl;\n    this.apiKey = apiKey;\n    this.metadata = metadata;\n    this.tenant = tenant;\n    this.language = language;\n    this.useWs = useWs;\n    // Try to load a saved conversation for this apiKey from localStorage\n    this.loadSavedConversation();\n  }\n\n  private getStorageKey(): string {\n    // Pointer to current conversation metadata for this apiKey\n    return `${this.storageKeyBase}:${this.apiKey}`;\n  }\n\n  /**\n   * Persist agent_chat_input_metadata to localStorage and dispatch a global event.\n   * Any consumer (e.g. config panel) can listen for GENASSIST_AGENT_METADATA_UPDATED\n   * or read from localStorage on mount; no need to pass callbacks into the chat.\n   */\n  private persistAndEmitAgentMetadata(metadata: Record<string, unknown>): void {\n    try {\n      const key = `genassist_agent_chat_input_metadata:${this.apiKey}`;\n      localStorage.setItem(key, JSON.stringify(metadata));\n      if (typeof window !== \"undefined\" && window.dispatchEvent) {\n        window.dispatchEvent(\n          new CustomEvent(GENASSIST_AGENT_METADATA_UPDATED, {\n            detail: { apiKey: this.apiKey, metadata: { ...metadata } },\n          })\n        );\n      }\n    } catch {\n      // ignore\n    }\n  }\n\n  /**\n   * Set the language for Accept-Language header\n   */\n  setLanguage(language: string | undefined): void {\n    this.language = language;\n  }\n\n  /**\n   * Update metadata (useful for updating language or other dynamic metadata)\n   */\n  setMetadata(metadata: Record<string, any> | undefined): void {\n    this.metadata = metadata;\n  }\n\n  /**\n   * Format language code for Accept-Language header\n   * Converts simple language codes (e.g., \"en\", \"es\") to proper format (e.g., \"en-US,en;q=0.9\")\n   * Follows HTTP Accept-Language header standards\n   */\n  private formatAcceptLanguage(langCode: string): string {\n    if (!langCode) return '';\n    \n    // Normalize language code (lowercase, handle common formats)\n    const normalized = langCode.toLowerCase().trim();\n    \n    // Map common language codes to their full locale format\n    const languageMap: Record<string, string> = {\n      'en': 'en-US',\n      'es': 'es-ES',\n      'fr': 'fr-FR',\n      'de': 'de-DE',\n      'it': 'it-IT',\n      'pt': 'pt-PT',\n      'ar': 'ar-SA',\n      'sq': 'sq-AL',\n    };\n    \n    // Check if the code already has a region (e.g., \"en-US\", \"es-ES\")\n    const hasRegion = normalized.includes('-');\n    \n    // Get the full locale or use the provided code if it already has a region\n    const fullLocale = hasRegion ? normalized : (languageMap[normalized] || normalized);\n    \n    // Extract base language from the full locale (e.g., \"en\" from \"en-US\")\n    const fullBaseLang = fullLocale.split('-')[0].toLowerCase();\n    \n    // Build Accept-Language header: primary locale, base language with quality, English fallback\n    // Format: \"locale,base-lang;q=0.9,en;q=0.8\"\n    const parts: string[] = [fullLocale];\n    \n    // fallback\n    parts.push(`${fullBaseLang};q=0.9`);\n    \n    // Add English as fallback if not already the primary language\n    if (fullBaseLang !== 'en') {\n      parts.push('en;q=0.8');\n    }\n    \n    return parts.join(', ');\n  }\n\n  /**\n   * Get headers for API requests, including authorization if guest token is available\n   */\n  private getHeaders(contentType: string = \"application/json\"): Record<string, string> {\n    const headers: Record<string, string> = {\n      \"x-api-key\": this.apiKey,\n      \"Content-Type\": contentType,\n    };\n\n    // Add tenant header if provided\n    if (this.tenant) {\n      headers[\"x-tenant-id\"] = this.tenant;\n    }\n\n    // Add authorization header if guest token is available\n    if (this.guestToken) {\n      headers[\"Authorization\"] = `Bearer ${this.guestToken}`;\n    }\n\n    // Add Accept-Language header if language is set\n    if (this.language) {\n      const acceptLanguage = this.formatAcceptLanguage(this.language);\n      if (acceptLanguage) {\n        headers[\"Accept-Language\"] = acceptLanguage;\n      }\n    }\n\n    return headers;\n  }\n\n  setMessageHandler(handler: (message: ChatMessage) => void) {\n    this.messageHandler = handler;\n  }\n\n  setTakeoverHandler(handler: () => void) {\n    this.takeoverHandler = handler;\n  }\n\n  setFinalizedHandler(handler: () => void) {\n    this.finalizedHandler = handler;\n  }\n\n  setConnectionStateHandler(\n    handler: (state: \"connecting\" | \"connected\" | \"disconnected\") => void\n  ) {\n    this.connectionStateHandler = handler;\n  }\n\n  setWelcomeDataHandler(handler: ((data: AgentWelcomeData) => void) | null) {\n    this.welcomeDataHandler = handler;\n  }\n\n  setServerUnavailableConfig(\n    message?: string,\n    contactUrl?: string,\n    contactLabel?: string\n  ): void {\n    this.serverUnavailableMessage = message;\n    this.serverUnavailableContactUrl = contactUrl;\n    this.serverUnavailableContactLabel = contactLabel;\n  }\n\n  getPossibleQueries(): string[] {\n    return this.possibleQueries;\n  }\n\n  getWelcomeData(): AgentWelcomeData {\n    return this.welcomeData;\n  }\n\n  getThinkingConfig(): AgentThinkingConfig {\n    return this.thinkingConfig;\n  }\n\n  /**\n   * Metadata keys/defaults from the workflow's Chat Input node (returned when conversation starts).\n   * Used to show workflow-defined metadata in the config panel.\n   */\n  getChatInputMetadata(): Record<string, unknown> {\n    return { ...this.chatInputMetadata };\n  }\n\n  /**\n   * Load a saved conversation ID from localStorage\n   */\n  private loadSavedConversation(): void {\n    try {\n      let savedConversation = localStorage.getItem(this.getStorageKey());\n      // Backward-compat: check old unscoped key if scoped one missing\n      if (!savedConversation) {\n        savedConversation = localStorage.getItem(this.storageKeyBase);\n      }\n      if (savedConversation) {\n        const {\n          conversationId,\n          createTime,\n          isFinalized,\n          possibleQueries,\n          welcomeData,\n          thinkingConfig,\n          agentId,\n          guestToken,\n        } = JSON.parse(savedConversation);\n        this.conversationId = conversationId;\n        this.conversationCreateTime = createTime;\n        this.isFinalized = isFinalized || false;\n        this.guestToken = guestToken || null;\n        this.possibleQueries = Array.isArray(possibleQueries)\n          ? possibleQueries\n          : [];\n        this.welcomeData = welcomeData || {};\n        if (!this.welcomeData.possibleQueries) {\n          this.welcomeData.possibleQueries = this.possibleQueries;\n        }\n        this.thinkingConfig = {\n          phrases: (thinkingConfig && thinkingConfig.phrases) || [],\n          delayMs:\n            thinkingConfig && typeof thinkingConfig.delayMs === \"number\"\n              ? thinkingConfig.delayMs\n              : 1000,\n        };\n        this.agentId = agentId;\n        if (!this.welcomeData.imageUrl && this.agentId) {\n          this.fetchWelcomeImage(this.agentId);\n        }\n        if (this.welcomeDataHandler) {\n          this.welcomeDataHandler(this.welcomeData);\n        }\n      }\n    } catch (error) {\n      // ignore\n    }\n  }\n\n  /**\n   * Save the current conversation ID to localStorage\n   */\n  private saveConversation(): void {\n    try {\n      if (this.conversationId && this.conversationCreateTime) {\n        const conversationData = {\n          conversationId: this.conversationId,\n          createTime: this.conversationCreateTime,\n          isFinalized: this.isFinalized,\n          possibleQueries: this.possibleQueries,\n          welcomeData: {\n            ...this.welcomeData,\n            imageUrl:\n              this.welcomeObjectUrl &&\n              this.welcomeData.imageUrl &&\n              this.welcomeData.imageUrl.startsWith(\"blob:\")\n                ? null\n                : this.welcomeData.imageUrl || null,\n          },\n          thinkingConfig: this.thinkingConfig,\n          agentId: this.agentId,\n          guestToken: this.guestToken,\n        };\n        localStorage.setItem(this.getStorageKey(), JSON.stringify(conversationData));\n      }\n    } catch (error) {\n      // ignore\n    }\n  }\n\n  // Check if an error is a token expiration error\n  private isTokenExpiredError(error: any): boolean {\n    return (\n      error.response &&\n      error.response.status === 401 &&\n      error.response.data &&\n      (error.response.data.error === \"Token has expired.\" ||\n        error.response.data.message === \"Token has expired.\" ||\n        (typeof error.response.data === \"string\" && error.response.data.includes(\"Token has expired\")))\n    );\n  }\n\n  /**\n   * Reset the current conversation by clearing the ID and websocket\n   */\n  resetChatConversation(): void {\n    // Close the current websocket connection if it exists\n    if (this.webSocket) {\n      this.webSocket.close();\n      this.webSocket = null;\n    }\n\n    // Clear the conversation ID\n    this.conversationId = null;\n    this.conversationCreateTime = null;\n    this.guestToken = null;\n    this.isFinalized = false;\n\n    // Clear possible queries and workflow metadata\n    this.possibleQueries = [];\n    this.welcomeData = {};\n    this.thinkingConfig = { phrases: [], delayMs: 1000 };\n    this.chatInputMetadata = {};\n    if (this.welcomeObjectUrl) {\n      try {\n        URL.revokeObjectURL(this.welcomeObjectUrl);\n      } catch {}\n      this.welcomeObjectUrl = null;\n    }\n    this.agentId = undefined;\n\n    // Remove from local storage\n    try {\n      localStorage.removeItem(this.getStorageKey());\n    } catch (error) {\n      // ignore\n    }\n  }\n\n  /**\n   * Check if there's a current conversation\n   */\n  hasActiveConversation(): boolean {\n    return !!this.conversationId;\n  }\n\n  /**\n   * Get the current conversation ID\n   */\n  getConversationId(): string | null {\n    return this.conversationId;\n  }\n\n  getConversationCreateTime(): number | null {\n    return this.conversationCreateTime;\n  }\n\n  isConversationFinalized(): boolean {\n    return this.isFinalized;\n  }\n\n  async startConversation(reCaptchaToken?: string | undefined): Promise<string> {\n    try {\n      const requestBody: any = {\n        messages: [],\n        recorded_at: new Date().toISOString(),\n        data_source_id: \"00000000-0000-0000-0000-000000000000\",\n      };\n\n      if (reCaptchaToken) {\n        requestBody.recaptcha_token = reCaptchaToken;\n      }\n\n      if (this.metadata) {\n        requestBody.metadata = this.metadata;\n      }\n\n      const response = await axios.post<StartConversationResponse>(\n        `${this.baseUrl}/api/conversations/in-progress/start`,\n        requestBody,\n        {\n          headers: this.getHeaders(),\n        }\n      );\n\n      this.conversationId = response.data.conversation_id;\n      // Store conversation create time (use from response if available, otherwise current time)\n      this.conversationCreateTime = response.data.create_time\n        ? response.data.create_time / 1000\n        : Date.now() / 1000;\n      // Store guest token if provided\n      this.guestToken = response.data.guest_token || null;\n      this.isFinalized = false;\n\n      // Store possible queries if available\n      if (\n        response.data.agent_possible_queries &&\n        response.data.agent_possible_queries.length > 0\n      ) {\n        this.possibleQueries = response.data.agent_possible_queries.filter(\n          (query) => typeof query === \"string\" && query.trim().length > 0\n        );\n      }\n\n      const anyData: any = response.data as any;\n      const agentId: string | undefined = anyData.agent_id;\n      this.agentId = agentId;\n      const rawMeta = anyData.agent_chat_input_metadata;\n      if (\n        rawMeta != null &&\n        typeof rawMeta === 'object' &&\n        !Array.isArray(rawMeta)\n      ) {\n        this.chatInputMetadata = { ...rawMeta };\n      } else {\n        this.chatInputMetadata = {};\n      }\n      this.persistAndEmitAgentMetadata(this.chatInputMetadata);\n      const welcomeTitle: string | undefined = anyData.agent_welcome_title;\n      const welcomeImageUrl: string | undefined =\n        anyData.agent_welcome_image_url;\n      const thinkingPhrases: string[] | undefined =\n        anyData.agent_thinking_phrases;\n      const thinkingDelaySec: number | undefined =\n        anyData.agent_thinking_phrase_delay;\n\n      this.welcomeData = {\n        title: welcomeTitle || null,\n        message: null,\n        imageUrl: welcomeImageUrl || null,\n        possibleQueries: this.possibleQueries,\n      };\n\n      if (Array.isArray(thinkingPhrases) && thinkingPhrases.length > 0) {\n        this.thinkingConfig.phrases = thinkingPhrases;\n      }\n      if (typeof thinkingDelaySec === \"number\" && thinkingDelaySec >= 0) {\n        this.thinkingConfig.delayMs = Math.max(\n          250,\n          Math.round(thinkingDelaySec * 1000)\n        );\n      }\n\n      if (!this.welcomeData.imageUrl && agentId) {\n        await this.fetchWelcomeImage(agentId);\n      }\n\n      // Process agent welcome message if available\n      if (response.data.agent_welcome_message && this.messageHandler) {\n        const now = Date.now() / 1000;\n        const welcomeMessage: ChatMessage = {\n          create_time: now,\n          start_time: now - this.conversationCreateTime, // Relative to conversation start\n          end_time: now - this.conversationCreateTime + 0.01, // Relative to conversation start\n          speaker: \"agent\",\n          text: response.data.agent_welcome_message,\n        };\n        this.welcomeData.message = response.data.agent_welcome_message;\n        this.messageHandler(welcomeMessage);\n      }\n\n      if (this.welcomeDataHandler) {\n        this.welcomeDataHandler(this.welcomeData);\n      }\n\n      this.saveConversation();\n      if (this.useWs) {\n        this.connectWebSocket();\n      }\n      return response.data.conversation_id;\n    } catch (error: any) {\n      if (this.isTokenExpiredError(error)) {\n        this.resetChatConversation();\n      }\n      throw error;\n    }\n  }\n\n  async sendMessage(\n    message: string,\n    attachments?: Attachment[],\n    extraMetadata?: Record<string, any>,\n    reCaptchaToken?: string\n  ): Promise<void> {\n    if (!this.conversationId || !this.conversationCreateTime) {\n      throw new Error(\"Conversation not started\");\n    }\n\n    const now = Date.now() / 1000;\n    const chatMessage: ChatMessage = {\n      create_time: now,\n      start_time: now - this.conversationCreateTime, // Relative to conversation start\n      end_time: now - this.conversationCreateTime + 0.01, // Relative to conversation start\n      speaker: \"customer\",\n      text: message,\n      attachments: attachments,\n    };\n\n    if (this.messageHandler) {\n      this.messageHandler(chatMessage);\n    }\n\n    try {\n      const requestBody: any = {\n        messages: [chatMessage],\n        recorded_at: new Date().toISOString(),\n      };\n\n      if (reCaptchaToken) {\n        requestBody.recaptcha_token = reCaptchaToken;\n      }\n\n      // Include metadata\n      const mergedMetadata = {\n        ...(this.metadata || {}),\n        ...(extraMetadata || {}),\n      };\n      if (Object.keys(mergedMetadata).length > 0) {\n        requestBody.metadata = mergedMetadata;\n      }\n\n      const response = await axios.patch(\n        `${this.baseUrl}/api/conversations/in-progress/update/${this.conversationId}`,\n        requestBody,\n        {\n          headers: this.getHeaders(),\n        }\n      );\n\n      // If not using WebSocket, try to retrieve the response message from the update conversation response\n      if (!this.useWs && this.messageHandler) {\n        try {\n          const responseData = response.data as any;\n          \n          if (responseData.messages && Array.isArray(responseData.messages)) {\n            // Look for the latest agent message in the response\n            for (let i = responseData.messages.length - 1; i >= 0; i--) {\n              const messageData = responseData.messages[i];\n              \n              if (\n                messageData.speaker === \"agent\" &&\n                messageData.text &&\n                messageData.create_time !== undefined &&\n                messageData.start_time !== undefined &&\n                messageData.end_time !== undefined\n              ) {\n                const agentMessage: ChatMessage = {\n                  create_time: messageData.create_time,\n                  start_time: this.conversationCreateTime\n                    ? messageData.start_time - this.conversationCreateTime\n                    : messageData.start_time,\n                  end_time: this.conversationCreateTime\n                    ? messageData.end_time - this.conversationCreateTime\n                    : messageData.end_time,\n                  speaker: \"agent\",\n                  text: messageData.text,\n                  message_id: messageData.message_id || messageData.id,\n                };\n                \n                // Only process if this is a new message we haven't seen before\n                // We can't easily check here, so we'll let the handler manage duplicates\n                this.messageHandler(agentMessage);\n                break;\n              }\n            }\n          }\n        } catch (parseError) {\n          console.error(\"Failed to parse update conversation response:\", parseError);\n        }\n      }\n    } catch (error: any) {\n      // Check if this is a token expiration error\n      if (this.isTokenExpiredError(error)) {\n        this.resetChatConversation();\n        throw error;\n      }\n\n      // Check if this is the agent inactive error\n      if (\n        error.response &&\n        error.response.data &&\n        error.response.data.error_key === \"AGENT_INACTIVE\"\n      ) {\n        // Create a custom message for the agent inactive error\n        if (this.messageHandler) {\n          const errorMessage: ChatMessage = {\n            create_time: now,\n            start_time: now - this.conversationCreateTime,\n            end_time: now - this.conversationCreateTime + 0.01,\n            speaker: \"special\",\n            text: this.serverUnavailableMessage ?? \"The agent is currently offline, please check back later. Thank you!\",\n            ...(this.serverUnavailableContactUrl && {\n              linkUrl: this.serverUnavailableContactUrl,\n              linkLabel: this.serverUnavailableContactLabel ?? \"Contact support\",\n            }),\n          };\n          this.messageHandler(errorMessage);\n        }\n        // Don't throw the error since we handled it with a message\n        return;\n      }\n\n      throw error;\n    }\n  }\n\n  async uploadFile(chatId: string, file: File): Promise<FileUploadResponse | null> {\n    if (!this.conversationId) {\n      throw new Error(\"Conversation not started\");\n    }\n\n    const formData = new FormData();\n    formData.append(\"chat_id\", chatId);\n    formData.append(\"file\", file);\n\n    try {\n      const response = await axios.post<FileUploadResponse>(\n        `${this.baseUrl}/api/genagent/knowledge/upload-chat-file`,\n        formData,\n        {\n          headers: this.getHeaders(\"multipart/form-data\"),\n        }\n      );\n\n      return response.data as FileUploadResponse;\n    } catch (error: any) {\n      if (this.isTokenExpiredError(error)) {\n        this.resetChatConversation();\n      }\n      throw error;\n    }\n  }\n\n  connectWebSocket(): void {\n    if (!this.useWs) {\n      return;\n    }\n\n    if (this.webSocket) {\n      this.webSocket.close();\n    }\n\n    if (!this.conversationId) {\n      throw new Error(\"Conversation ID is required for WebSocket connection\");\n    }\n\n    if (this.connectionStateHandler) this.connectionStateHandler(\"connecting\");\n    \n    // Build WebSocket URL with proper authentication\n    const wsBase = this.baseUrl.replace(\"http\", \"ws\");\n    const topics = [\"message\", \"takeover\", \"finalize\"];\n    const topicsQuery = topics.map((t) => `topics=${t}`).join(\"&\");\n    \n    // Use guest_token if available, otherwise fall back to api_key\n    const authParam = this.guestToken \n      ? `access_token=${encodeURIComponent(this.guestToken)}`\n      : `api_key=${encodeURIComponent(this.apiKey)}`;\n    \n    let wsUrl = `${wsBase}/api/conversations/ws/${this.conversationId}?${authParam}&lang=en&${topicsQuery}`;\n    \n    // Add tenant as query parameter if provided\n    if (this.tenant) {\n      wsUrl += `&x-tenant-id=${encodeURIComponent(this.tenant)}`;\n    }\n    \n    // Use native browser WebSocket factory\n    this.webSocket = createWebSocket(wsUrl);\n\n\n    this.webSocket.onopen = () => {\n      if (this.connectionStateHandler) this.connectionStateHandler(\"connected\");\n    };\n\n    this.webSocket.onmessage = (event: MessageEvent) => {\n      try {\n        const data = JSON.parse(event.data as string);\n\n        if (data.type === \"message\" && this.messageHandler) {\n          if (Array.isArray(data.payload)) {\n            const messages = data.payload as ChatMessage[];\n            // Adjust timestamps to be relative to conversation start\n            // Also normalize message_id field (backend might send 'id' instead of 'message_id')\n            const adjustedMessages = messages\n              .map((msg) => {\n                const adjusted = this.adjustMessageTimestamps(msg);\n                // Normalize: if backend sends 'id', use it as 'message_id'\n                if (!adjusted.message_id && (msg as any).id) {\n                  adjusted.message_id = (msg as any).id;\n                }\n                return adjusted;\n              })\n              .filter((msg) => msg.speaker !== \"customer\");\n            adjustedMessages.forEach(this.messageHandler);\n          } else {\n            const adjustedMessage = this.adjustMessageTimestamps(\n              data.payload as ChatMessage\n            );\n            // Normalize: if backend sends 'id', use it as 'message_id'\n            if (!adjustedMessage.message_id && (data.payload as any).id) {\n              adjustedMessage.message_id = (data.payload as any).id;\n            }\n            if (adjustedMessage.speaker !== \"customer\") {\n              this.messageHandler(adjustedMessage);\n            }\n          }\n        } else if (data.type === \"takeover\") {\n          // Handle takeover event\n          // Create special message for the takeover indicator\n          if (this.messageHandler) {\n            const now = Date.now() / 1000;\n            const takeoverMessage: ChatMessage = {\n              create_time: now,\n              start_time: this.conversationCreateTime\n                ? now - this.conversationCreateTime\n                : 0,\n              end_time: this.conversationCreateTime\n                ? now - this.conversationCreateTime + 0.01\n                : 0.01,\n              speaker: \"special\",\n              text: \"Supervisor took over\",\n            };\n            this.messageHandler(takeoverMessage);\n          }\n\n          // Call the takeover handler if provided\n          if (this.takeoverHandler) {\n            this.takeoverHandler();\n          }\n        } else if (data.type === \"finalize\") {\n          // Handle finalized event\n          // Create special message for the finalized indicator\n          if (this.messageHandler) {\n            const now = Date.now() / 1000;\n            const finalizedMessage: ChatMessage = {\n              create_time: now,\n              start_time: this.conversationCreateTime\n                ? now - this.conversationCreateTime\n                : 0,\n              end_time: this.conversationCreateTime\n                ? now - this.conversationCreateTime + 0.01\n                : 0.01,\n              speaker: \"special\",\n              text: \"Conversation Finalized\",\n            };\n            this.messageHandler(finalizedMessage);\n          }\n\n          // Call the finalized handler if provided\n          if (this.finalizedHandler) {\n            this.finalizedHandler();\n          }\n          this.isFinalized = true;\n          this.saveConversation();\n        }\n      } catch (error) {\n        // ignore\n      }\n    };\n\n    this.webSocket.onerror = (error: Event) => {\n      if (this.connectionStateHandler)\n        this.connectionStateHandler(\"disconnected\");\n      \n      // Log diagnostic\n      const diagnostic = createWebSocketDiagnostic(error, wsUrl);\n      console.error(`[GenAssist Chat] ${diagnostic}`);\n    };\n\n    this.webSocket.onclose = (event: CloseEvent) => {\n      if (this.connectionStateHandler)\n        this.connectionStateHandler(\"disconnected\");\n      \n      // Log diagnostic\n      if (!event.wasClean) {\n        const diagnostic = createWebSocketDiagnostic(event, wsUrl);\n        console.warn(`[GenAssist Chat] ${diagnostic}`);\n      }\n    };\n  }\n\n  disconnect(): void {\n    if (this.webSocket) {\n      this.webSocket.close();\n      this.webSocket = null;\n    }\n  }\n\n  private async fetchWelcomeImage(agentId: string): Promise<void> {\n    try {\n      const imageResponse = await axios.get(\n        `${this.baseUrl}/api/genagent/agents/configs/${agentId}/welcome-image`,\n        {\n          headers: this.getHeaders(),\n          responseType: \"blob\",\n        }\n      );\n      const blobUrl = URL.createObjectURL(imageResponse.data);\n      this.welcomeObjectUrl = blobUrl;\n      this.welcomeData.imageUrl = blobUrl;\n      if (this.welcomeDataHandler) {\n        this.welcomeDataHandler(this.welcomeData);\n      }\n    } catch (err: any) {\n      if (this.isTokenExpiredError(err)) {\n        this.resetChatConversation();\n      }\n      // ignore\n    }\n  }\n\n  // Add feedback to a specific agent message in the conversation\n  async addFeedback(\n    messageId: string,\n    feedback: \"good\" | \"bad\",\n    feedback_message?: string\n  ): Promise<void> {\n    if (!messageId) {\n      throw new Error(\"Message ID is required for feedback\");\n    }\n\n    try {\n      // Use message ID in the URL path, not conversation ID\n      const url = `${this.baseUrl}/api/conversations/message/add-feedback/${messageId}`;\n      // Body should only contain feedback and feedback_message (no message_id)\n      const payload: {\n        feedback: \"good\" | \"bad\";\n        feedback_message?: string;\n      } = {\n        feedback,\n      };\n      \n      if (feedback_message) {\n        payload.feedback_message = feedback_message;\n      }\n      \n      await axios.patch(url, payload, {\n        headers: this.getHeaders(),\n      });\n      \n    } catch (error: any) {\n      if (this.isTokenExpiredError(error)) {\n        this.resetChatConversation();\n      }\n      console.error('Feedback API call failed:', {\n        message: error.message,\n        response: error.response?.data,\n        status: error.response?.status,\n      });\n      throw error;\n    }\n  }\n\n  // Helper method to adjust message timestamps relative to conversation start\n  private adjustMessageTimestamps(message: ChatMessage): ChatMessage {\n    if (!this.conversationCreateTime) {\n      return message;\n    }\n\n    return {\n      ...message,\n      start_time: message.start_time - this.conversationCreateTime,\n      end_time: message.end_time - this.conversationCreateTime,\n    };\n  }\n}\n"]}
|
|
908
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chatService.js","sourceRoot":"","sources":["../../src/services/chatService.ts"],"names":[],"mappings":"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","sourcesContent":["import axios from \"axios\";\nimport {\n  ChatMessage,\n  StartConversationResponse,\n  Attachment,\n  AgentThinkingConfig,\n  AgentWelcomeData,\n  FileUploadResponse,\n  InProgressPollResponse,\n  InProgressPollMessage,\n} from \"../types\";\nimport {\n  createWebSocket,\n  createWebSocketDiagnostic,\n} from \"../utils/websocket\";\nexport const GENASSIST_AGENT_METADATA_UPDATED = \"genassist_agent_metadata_updated\";\n\nexport class ChatService {\n  private baseUrl: string;\n  private apiKey: string;\n  private metadata: Record<string, any> | undefined;\n  private conversationId: string | null = null;\n  private conversationCreateTime: number | null = null; // Track conversation start time\n  private guestToken: string | null = null; // Guest token for authorization\n  private isFinalized: boolean = false;\n  private webSocket: WebSocket | null = null;\n  private messageHandler: ((message: ChatMessage) => void) | null = null;\n  private takeoverHandler: (() => void) | null = null;\n  private finalizedHandler: (() => void) | null = null;\n  private connectionStateHandler:\n    | ((state: \"connecting\" | \"connected\" | \"disconnected\") => void)\n    | null = null;\n  private welcomeDataHandler: ((data: AgentWelcomeData) => void) | null = null;\n  private storageKeyBase = \"genassist_conversation\";\n  private possibleQueries: string[] = [];\n  private welcomeData: AgentWelcomeData = {};\n  private thinkingConfig: AgentThinkingConfig = { phrases: [], delayMs: 1000 };\n  private chatInputMetadata: Record<string, unknown> = {};\n  private welcomeObjectUrl: string | null = null; // to revoke on reset\n  private tenant: string | undefined;\n  private agentId: string | undefined;\n  private language: string | undefined;\n  private useWs: boolean;\n  private serverUnavailableMessage: string | undefined;\n  private serverUnavailableContactUrl: string | undefined;\n  private serverUnavailableContactLabel: string | undefined;\n  private usePoll: boolean = false;\n\n  constructor(\n    baseUrl: string,\n    apiKey: string,\n    metadata?: Record<string, any>,\n    tenant?: string,\n    language?: string,\n    useWs: boolean = true,\n    usePoll: boolean = false\n  ) {\n    this.baseUrl = baseUrl.endsWith(\"/\") ? baseUrl.slice(0, -1) : baseUrl;\n    this.apiKey = apiKey;\n    this.metadata = metadata;\n    this.tenant = tenant;\n    this.language = language;\n    this.useWs = useWs;\n    this.usePoll = usePoll;\n    // Try to load a saved conversation for this apiKey from localStorage\n    this.loadSavedConversation();\n  }\n\n  private getStorageKey(): string {\n    // Pointer to current conversation metadata for this apiKey\n    return `${this.storageKeyBase}:${this.apiKey}`;\n  }\n\n  /**\n   * Handle conversation finalized: emit special message, call finalized handler, persist state.\n   */\n  public handleConversationFinalized(): void {\n    if (this.messageHandler) {\n      const now = Date.now() / 1000;\n      const finalizedMessage: ChatMessage = {\n        create_time: now,\n        start_time: this.conversationCreateTime\n          ? now - this.conversationCreateTime\n          : 0,\n        end_time: this.conversationCreateTime\n          ? now - this.conversationCreateTime + 0.01\n          : 0.01,\n        speaker: \"special\",\n        text: \"Conversation Finalized\",\n        type: \"finalized\",\n      };\n      this.messageHandler(finalizedMessage);\n    }\n    if (this.finalizedHandler) {\n      this.finalizedHandler();\n    }\n    this.isFinalized = true;\n    this.saveConversation();\n  }\n\n  /**\n   * Persist agent_chat_input_metadata to localStorage and dispatch a global event.\n   * Any consumer (e.g. config panel) can listen for GENASSIST_AGENT_METADATA_UPDATED\n   * or read from localStorage on mount; no need to pass callbacks into the chat.\n   */\n  private persistAndEmitAgentMetadata(metadata: Record<string, unknown>): void {\n    try {\n      const key = `genassist_agent_chat_input_metadata:${this.apiKey}`;\n      localStorage.setItem(key, JSON.stringify(metadata));\n      if (typeof window !== \"undefined\" && window.dispatchEvent) {\n        window.dispatchEvent(\n          new CustomEvent(GENASSIST_AGENT_METADATA_UPDATED, {\n            detail: { apiKey: this.apiKey, metadata: { ...metadata } },\n          })\n        );\n      }\n    } catch {\n      // ignore\n    }\n  }\n\n  /**\n   * Set the language for Accept-Language header\n   */\n  setLanguage(language: string | undefined): void {\n    this.language = language;\n  }\n\n  /**\n   * Update metadata (useful for updating language or other dynamic metadata)\n   */\n  setMetadata(metadata: Record<string, any> | undefined): void {\n    this.metadata = metadata;\n  }\n\n  /**\n   * Format language code for Accept-Language header\n   * Converts simple language codes (e.g., \"en\", \"es\") to proper format (e.g., \"en-US,en;q=0.9\")\n   * Follows HTTP Accept-Language header standards\n   */\n  private formatAcceptLanguage(langCode: string): string {\n    if (!langCode) return '';\n\n    // Normalize language code (lowercase, handle common formats)\n    const normalized = langCode.toLowerCase().trim();\n\n    // Map common language codes to their full locale format\n    const languageMap: Record<string, string> = {\n      'en': 'en-US',\n      'es': 'es-ES',\n      'fr': 'fr-FR',\n      'de': 'de-DE',\n      'it': 'it-IT',\n      'pt': 'pt-PT',\n      'ar': 'ar-SA',\n      'sq': 'sq-AL',\n    };\n\n    // Check if the code already has a region (e.g., \"en-US\", \"es-ES\")\n    const hasRegion = normalized.includes('-');\n\n    // Get the full locale or use the provided code if it already has a region\n    const fullLocale = hasRegion ? normalized : (languageMap[normalized] || normalized);\n\n    // Extract base language from the full locale (e.g., \"en\" from \"en-US\")\n    const fullBaseLang = fullLocale.split('-')[0].toLowerCase();\n\n    // Build Accept-Language header: primary locale, base language with quality, English fallback\n    // Format: \"locale,base-lang;q=0.9,en;q=0.8\"\n    const parts: string[] = [fullLocale];\n\n    // fallback\n    parts.push(`${fullBaseLang};q=0.9`);\n\n    // Add English as fallback if not already the primary language\n    if (fullBaseLang !== 'en') {\n      parts.push('en;q=0.8');\n    }\n\n    return parts.join(', ');\n  }\n\n  /**\n   * Get headers for API requests, including authorization if guest token is available\n   */\n  private getHeaders(contentType: string = \"application/json\"): Record<string, string> {\n    const headers: Record<string, string> = {\n      \"x-api-key\": this.apiKey,\n      \"Content-Type\": contentType,\n    };\n\n    // Add tenant header if provided\n    if (this.tenant) {\n      headers[\"x-tenant-id\"] = this.tenant;\n    }\n\n    // Add authorization header if guest token is available\n    if (this.guestToken) {\n      headers[\"Authorization\"] = `Bearer ${this.guestToken}`;\n    }\n\n    // Add Accept-Language header if language is set\n    if (this.language) {\n      const acceptLanguage = this.formatAcceptLanguage(this.language);\n      if (acceptLanguage) {\n        headers[\"Accept-Language\"] = acceptLanguage;\n      }\n    }\n\n    return headers;\n  }\n\n  setMessageHandler(handler: (message: ChatMessage) => void) {\n    this.messageHandler = handler;\n  }\n\n  setTakeoverHandler(handler: () => void) {\n    this.takeoverHandler = handler;\n  }\n\n  setFinalizedHandler(handler: () => void) {\n    this.finalizedHandler = handler;\n  }\n\n  setConnectionStateHandler(\n    handler: (state: \"connecting\" | \"connected\" | \"disconnected\") => void\n  ) {\n    this.connectionStateHandler = handler;\n  }\n\n  setWelcomeDataHandler(handler: ((data: AgentWelcomeData) => void) | null) {\n    this.welcomeDataHandler = handler;\n  }\n\n  setServerUnavailableConfig(\n    message?: string,\n    contactUrl?: string,\n    contactLabel?: string\n  ): void {\n    this.serverUnavailableMessage = message;\n    this.serverUnavailableContactUrl = contactUrl;\n    this.serverUnavailableContactLabel = contactLabel;\n  }\n\n  getPossibleQueries(): string[] {\n    return this.possibleQueries;\n  }\n\n  getWelcomeData(): AgentWelcomeData {\n    return this.welcomeData;\n  }\n\n  getThinkingConfig(): AgentThinkingConfig {\n    return this.thinkingConfig;\n  }\n\n  /**\n   * Metadata keys/defaults from the workflow's Chat Input node (returned when conversation starts).\n   * Used to show workflow-defined metadata in the config panel.\n   */\n  getChatInputMetadata(): Record<string, unknown> {\n    return { ...this.chatInputMetadata };\n  }\n\n  /**\n   * Load a saved conversation ID from localStorage\n   */\n  private loadSavedConversation(): void {\n    try {\n      let savedConversation = localStorage.getItem(this.getStorageKey());\n      // Backward-compat: check old unscoped key if scoped one missing\n      if (!savedConversation) {\n        savedConversation = localStorage.getItem(this.storageKeyBase);\n      }\n      if (savedConversation) {\n        const {\n          conversationId,\n          createTime,\n          isFinalized,\n          possibleQueries,\n          welcomeData,\n          thinkingConfig,\n          agentId,\n          guestToken,\n        } = JSON.parse(savedConversation);\n        this.conversationId = conversationId;\n        this.conversationCreateTime = createTime;\n        this.isFinalized = isFinalized || false;\n        this.guestToken = guestToken || null;\n        this.possibleQueries = Array.isArray(possibleQueries)\n          ? possibleQueries\n          : [];\n        this.welcomeData = welcomeData || {};\n        if (!this.welcomeData.possibleQueries) {\n          this.welcomeData.possibleQueries = this.possibleQueries;\n        }\n        this.thinkingConfig = {\n          phrases: (thinkingConfig && thinkingConfig.phrases) || [],\n          delayMs:\n            thinkingConfig && typeof thinkingConfig.delayMs === \"number\"\n              ? thinkingConfig.delayMs\n              : 1000,\n        };\n        this.agentId = agentId;\n        if (!this.welcomeData.imageUrl && this.agentId) {\n          this.fetchWelcomeImage(this.agentId);\n        }\n        if (this.welcomeDataHandler) {\n          this.welcomeDataHandler(this.welcomeData);\n        }\n      }\n    } catch (error) {\n      // ignore\n    }\n  }\n\n  /**\n   * Save the current conversation ID to localStorage\n   */\n  private saveConversation(): void {\n    try {\n      if (this.conversationId && this.conversationCreateTime) {\n        const conversationData = {\n          conversationId: this.conversationId,\n          createTime: this.conversationCreateTime,\n          isFinalized: this.isFinalized,\n          possibleQueries: this.possibleQueries,\n          welcomeData: {\n            ...this.welcomeData,\n            imageUrl:\n              this.welcomeObjectUrl &&\n              this.welcomeData.imageUrl &&\n              this.welcomeData.imageUrl.startsWith(\"blob:\")\n                ? null\n                : this.welcomeData.imageUrl || null,\n          },\n          thinkingConfig: this.thinkingConfig,\n          agentId: this.agentId,\n          guestToken: this.guestToken,\n        };\n        localStorage.setItem(this.getStorageKey(), JSON.stringify(conversationData));\n      }\n    } catch (error) {\n      // ignore\n    }\n  }\n\n  // Check if an error is a token expiration error\n  private isTokenExpiredError(error: any): boolean {\n    return (\n      error.response &&\n      error.response.status === 401 &&\n      error.response.data &&\n      (error.response.data.error === \"Token has expired.\" ||\n        error.response.data.message === \"Token has expired.\" ||\n        (typeof error.response.data === \"string\" && error.response.data.includes(\"Token has expired\")))\n    );\n  }\n\n  /**\n   * Internal handler for takeover events: emits a special message and invokes takeoverHandler.\n   * Runs only once per conversation (idempotent).\n   */\n  private handleTakeover(now?: number): void {\n    const t = now ?? Date.now() / 1000;\n    if (this.messageHandler) {\n      const takeoverMessage: ChatMessage = {\n        create_time: t,\n        start_time: this.conversationCreateTime ? t - this.conversationCreateTime : 0,\n        end_time: this.conversationCreateTime ? t - this.conversationCreateTime + 0.01 : 0.01,\n        speaker: \"special\",\n        text: \"Supervisor took over\",\n        type: \"takeover\",\n      };\n\n      this.messageHandler(takeoverMessage);\n    }\n    if (this.takeoverHandler) {\n      this.takeoverHandler();\n    }\n  }\n\n  /**\n   * Reset the current conversation by clearing the ID and websocket\n   */\n  resetChatConversation(): void {\n    // Close the current websocket connection if it exists\n    if (this.webSocket) {\n      this.webSocket.close();\n      this.webSocket = null;\n    }\n\n    // Clear the conversation ID\n    this.conversationId = null;\n    this.conversationCreateTime = null;\n    this.guestToken = null;\n    this.isFinalized = false;\n\n    // Clear possible queries and workflow metadata\n    this.possibleQueries = [];\n    this.welcomeData = {};\n    this.thinkingConfig = { phrases: [], delayMs: 1000 };\n    this.chatInputMetadata = {};\n    if (this.welcomeObjectUrl) {\n      try {\n        URL.revokeObjectURL(this.welcomeObjectUrl);\n      } catch {}\n      this.welcomeObjectUrl = null;\n    }\n    this.agentId = undefined;\n\n    // Remove from local storage\n    try {\n      localStorage.removeItem(this.getStorageKey());\n    } catch (error) {\n      // ignore\n    }\n  }\n\n  /**\n   * Check if there's a current conversation\n   */\n  hasActiveConversation(): boolean {\n    return !!this.conversationId;\n  }\n\n  /**\n   * Get the current conversation ID\n   */\n  getConversationId(): string | null {\n    return this.conversationId;\n  }\n\n  getConversationCreateTime(): number | null {\n    return this.conversationCreateTime;\n  }\n\n  isConversationFinalized(): boolean {\n    return this.isFinalized;\n  }\n\n  async startConversation(reCaptchaToken?: string | undefined): Promise<string> {\n    try {\n      const requestBody: any = {\n        messages: [],\n        recorded_at: new Date().toISOString(),\n        data_source_id: \"00000000-0000-0000-0000-000000000000\",\n      };\n\n      if (reCaptchaToken) {\n        requestBody.recaptcha_token = reCaptchaToken;\n      }\n\n      if (this.metadata) {\n        requestBody.metadata = this.metadata;\n      }\n\n      const response = await axios.post<StartConversationResponse>(\n        `${this.baseUrl}/api/conversations/in-progress/start`,\n        requestBody,\n        {\n          headers: this.getHeaders(),\n        }\n      );\n\n      this.conversationId = response.data.conversation_id;\n      // Store conversation create time (use from response if available, otherwise current time)\n      this.conversationCreateTime = response.data.create_time\n        ? response.data.create_time / 1000\n        : Date.now() / 1000;\n      // Store guest token if provided\n      this.guestToken = response.data.guest_token || null;\n      this.isFinalized = false;\n\n      // Store possible queries if available\n      if (\n        response.data.agent_possible_queries &&\n        response.data.agent_possible_queries.length > 0\n      ) {\n        this.possibleQueries = response.data.agent_possible_queries.filter(\n          (query) => typeof query === \"string\" && query.trim().length > 0\n        );\n      }\n\n      const anyData: any = response.data as any;\n      const agentId: string | undefined = anyData.agent_id;\n      this.agentId = agentId;\n      const rawMeta = anyData.agent_chat_input_metadata;\n      if (\n        rawMeta != null &&\n        typeof rawMeta === 'object' &&\n        !Array.isArray(rawMeta)\n      ) {\n        this.chatInputMetadata = { ...rawMeta };\n      } else {\n        this.chatInputMetadata = {};\n      }\n      this.persistAndEmitAgentMetadata(this.chatInputMetadata);\n      const welcomeTitle: string | undefined = anyData.agent_welcome_title;\n      const welcomeImageUrl: string | undefined =\n        anyData.agent_welcome_image_url;\n      const thinkingPhrases: string[] | undefined =\n        anyData.agent_thinking_phrases;\n      const thinkingDelaySec: number | undefined =\n        anyData.agent_thinking_phrase_delay;\n\n      this.welcomeData = {\n        title: welcomeTitle || null,\n        message: null,\n        imageUrl: welcomeImageUrl || null,\n        possibleQueries: this.possibleQueries,\n      };\n\n      if (Array.isArray(thinkingPhrases) && thinkingPhrases.length > 0) {\n        this.thinkingConfig.phrases = thinkingPhrases;\n      }\n      if (typeof thinkingDelaySec === \"number\" && thinkingDelaySec >= 0) {\n        this.thinkingConfig.delayMs = Math.max(\n          250,\n          Math.round(thinkingDelaySec * 1000)\n        );\n      }\n\n      if (!this.welcomeData.imageUrl && agentId) {\n        await this.fetchWelcomeImage(agentId);\n      }\n\n      // Process agent welcome message if available\n      if (response.data.agent_welcome_message && this.messageHandler) {\n        const now = Date.now() / 1000;\n        const welcomeMessage: ChatMessage = {\n          create_time: now,\n          start_time: now - this.conversationCreateTime, // Relative to conversation start\n          end_time: now - this.conversationCreateTime + 0.01, // Relative to conversation start\n          speaker: \"agent\",\n          text: response.data.agent_welcome_message,\n        };\n        this.welcomeData.message = response.data.agent_welcome_message;\n        this.messageHandler(welcomeMessage);\n      }\n\n      if (this.welcomeDataHandler) {\n        this.welcomeDataHandler(this.welcomeData);\n      }\n\n      this.saveConversation();\n      if (this.useWs) {\n        this.connectWebSocket();\n      }\n      return response.data.conversation_id;\n    } catch (error: any) {\n      if (this.isTokenExpiredError(error)) {\n        this.resetChatConversation();\n      }\n      throw error;\n    }\n  }\n\n  async sendMessage(\n    message: string,\n    attachments?: Attachment[],\n    extraMetadata?: Record<string, any>,\n    reCaptchaToken?: string\n  ): Promise<void> {\n    if (!this.conversationId || !this.conversationCreateTime) {\n      throw new Error(\"Conversation not started\");\n    }\n\n    const now = Date.now() / 1000;\n    const chatMessage: ChatMessage = {\n      create_time: now,\n      start_time: now - this.conversationCreateTime, // Relative to conversation start\n      end_time: now - this.conversationCreateTime + 0.01, // Relative to conversation start\n      speaker: \"customer\",\n      text: message,\n      attachments: attachments,\n    };\n\n    if (this.messageHandler) {\n      this.messageHandler(chatMessage);\n    }\n\n    try {\n      const requestBody: any = {\n        messages: [chatMessage],\n        recorded_at: new Date().toISOString(),\n      };\n\n      if (reCaptchaToken) {\n        requestBody.recaptcha_token = reCaptchaToken;\n      }\n\n      // Include metadata\n      const mergedMetadata = {\n        ...(this.metadata || {}),\n        ...(extraMetadata || {}),\n      };\n      if (Object.keys(mergedMetadata).length > 0) {\n        requestBody.metadata = mergedMetadata;\n      }\n\n      const response = await axios.patch(\n        `${this.baseUrl}/api/conversations/in-progress/update/${this.conversationId}`,\n        requestBody,\n        {\n          headers: this.getHeaders(),\n        }\n      ).catch(e => {\n        const err = e?.response?.data;\n        // Handle error when conversation is finalized\n        if (err?.error_key === 'CONVERSATION_FINALIZED') {\n          this.handleConversationFinalized();\n        } else {\n          throw e;\n        }\n      });\n\n      // If not using WebSocket, try to retrieve the response message from the update conversation response\n      if (!this.useWs && this.messageHandler && (!this.useWs || !this.usePoll)) { // usePoll is true when using heartbeat polling\n        try {\n          const responseData = response?.data as any || {};\n\n          if (responseData.messages && Array.isArray(responseData.messages)) {\n\n            // Look for the latest agent message in the response\n            for (let i = responseData.messages.length - 1; i >= 0; i--) {\n              const messageData = responseData.messages[i];\n\n              if (\n                messageData.speaker === \"agent\" &&\n                messageData.text &&\n                messageData.create_time !== undefined &&\n                messageData.start_time !== undefined &&\n                messageData.end_time !== undefined\n              ) {\n                const agentMessage: ChatMessage = {\n                  create_time: messageData.create_time,\n                  start_time: this.conversationCreateTime\n                    ? messageData.start_time - this.conversationCreateTime\n                    : messageData.start_time,\n                  end_time: this.conversationCreateTime\n                    ? messageData.end_time - this.conversationCreateTime\n                    : messageData.end_time,\n                  speaker: \"agent\",\n                  text: messageData.text,\n                  message_id: messageData.message_id || messageData.id,\n                };\n\n                // Only process if this is a new message we haven't seen before\n                // We can't easily check here, so we'll let the handler manage duplicates\n                this.messageHandler(agentMessage);\n                break;\n              }\n            }\n          }\n        } catch (parseError) {\n          console.error(\"Failed to parse update conversation response:\", parseError);\n        }\n      }\n    } catch (error: any) {\n      // Check if this is a token expiration error\n      if (this.isTokenExpiredError(error)) {\n        this.resetChatConversation();\n        throw error;\n      }\n\n      // Check if this is the agent inactive error\n      if (\n        error.response &&\n        error.response.data &&\n        error.response.data.error_key === \"AGENT_INACTIVE\"\n      ) {\n        // Create a custom message for the agent inactive error\n        if (this.messageHandler) {\n          const errorMessage: ChatMessage = {\n            create_time: now,\n            start_time: now - this.conversationCreateTime,\n            end_time: now - this.conversationCreateTime + 0.01,\n            speaker: \"special\",\n            text: this.serverUnavailableMessage ?? \"The agent is currently offline, please check back later. Thank you!\",\n            ...(this.serverUnavailableContactUrl && {\n              linkUrl: this.serverUnavailableContactUrl,\n              linkLabel: this.serverUnavailableContactLabel ?? \"Contact support\",\n            }),\n          };\n          this.messageHandler(errorMessage);\n        }\n        // Don't throw the error since we handled it with a message\n        return;\n      }\n\n      throw error;\n    }\n  }\n\n  async uploadFile(chatId: string, file: File): Promise<FileUploadResponse | null> {\n    if (!this.conversationId) {\n      throw new Error(\"Conversation not started\");\n    }\n\n    const formData = new FormData();\n    formData.append(\"chat_id\", chatId);\n    formData.append(\"file\", file);\n\n    try {\n      const response = await axios.post<FileUploadResponse>(\n        `${this.baseUrl}/api/genagent/knowledge/upload-chat-file`,\n        formData,\n        {\n          headers: this.getHeaders(\"multipart/form-data\"),\n        }\n      );\n\n      return response.data as FileUploadResponse;\n    } catch (error: any) {\n      if (this.isTokenExpiredError(error)) {\n        this.resetChatConversation();\n      }\n      throw error;\n    }\n  }\n\n  connectWebSocket(): void {\n    if (!this.useWs) {\n      return;\n    }\n\n    if (this.webSocket) {\n      this.webSocket.close();\n    }\n\n    if (!this.conversationId) {\n      throw new Error(\"Conversation ID is required for WebSocket connection\");\n    }\n\n    if (this.connectionStateHandler) this.connectionStateHandler(\"connecting\");\n\n    // Build WebSocket URL with proper authentication\n    const wsBase = this.baseUrl.replace(\"http\", \"ws\");\n    const topics = [\"message\", \"takeover\", \"finalize\"];\n    const topicsQuery = topics.map((t) => `topics=${t}`).join(\"&\");\n\n    // Use guest_token if available, otherwise fall back to api_key\n    const authParam = this.guestToken\n      ? `access_token=${encodeURIComponent(this.guestToken)}`\n      : `api_key=${encodeURIComponent(this.apiKey)}`;\n\n    let wsUrl = `${wsBase}/api/conversations/ws/${this.conversationId}?${authParam}&lang=en&${topicsQuery}`;\n\n    // Add tenant as query parameter if provided\n    if (this.tenant) {\n      wsUrl += `&x-tenant-id=${encodeURIComponent(this.tenant)}`;\n    }\n\n    // Use native browser WebSocket factory\n    this.webSocket = createWebSocket(wsUrl);\n\n\n    this.webSocket.onopen = () => {\n      if (this.connectionStateHandler) this.connectionStateHandler(\"connected\");\n    };\n\n    this.webSocket.onmessage = (event: MessageEvent) => {\n      try {\n        const data = JSON.parse(event.data as string);\n\n        if (data.type === \"message\" && this.messageHandler) {\n          if (Array.isArray(data.payload)) {\n            const messages = data.payload as ChatMessage[];\n            // Adjust timestamps to be relative to conversation start\n            // Also normalize message_id field (backend might send 'id' instead of 'message_id')\n            const adjustedMessages = messages\n              .map((msg) => {\n                const adjusted = this.adjustMessageTimestamps(msg);\n                // Normalize: if backend sends 'id', use it as 'message_id'\n                if (!adjusted.message_id && (msg as any).id) {\n                  adjusted.message_id = (msg as any).id;\n                }\n                return adjusted;\n              })\n              .filter((msg) => msg.speaker !== \"customer\");\n            adjustedMessages.forEach(this.messageHandler);\n          } else {\n            const adjustedMessage = this.adjustMessageTimestamps(\n              data.payload as ChatMessage\n            );\n            // Normalize: if backend sends 'id', use it as 'message_id'\n            if (!adjustedMessage.message_id && (data.payload as any).id) {\n              adjustedMessage.message_id = (data.payload as any).id;\n            }\n            if (adjustedMessage.speaker !== \"customer\") {\n              this.messageHandler(adjustedMessage);\n            }\n          }\n        } else if (data.type === \"takeover\") {\n          this.handleTakeover();\n        } else if (data.type === \"finalize\") {\n          this.handleConversationFinalized();\n        }\n      } catch (error) {\n        // ignore\n      }\n    };\n\n    this.webSocket.onerror = (error: Event) => {\n      if (this.connectionStateHandler)\n        this.connectionStateHandler(\"disconnected\");\n\n      // Log diagnostic\n      const diagnostic = createWebSocketDiagnostic(error, wsUrl);\n      console.error(`[GenAssist Chat] ${diagnostic}`);\n    };\n\n    this.webSocket.onclose = (event: CloseEvent) => {\n      if (this.connectionStateHandler)\n        this.connectionStateHandler(\"disconnected\");\n\n      // Log diagnostic\n      if (!event.wasClean) {\n        const diagnostic = createWebSocketDiagnostic(event, wsUrl);\n        console.warn(`[GenAssist Chat] ${diagnostic}`);\n      }\n    };\n  }\n\n  disconnect(): void {\n    if (this.webSocket) {\n      this.webSocket.close();\n      this.webSocket = null;\n    }\n  }\n\n  private async fetchWelcomeImage(agentId: string): Promise<void> {\n    try {\n      const imageResponse = await axios.get(\n        `${this.baseUrl}/api/genagent/agents/configs/${agentId}/welcome-image`,\n        {\n          headers: this.getHeaders(),\n          responseType: \"blob\",\n        }\n      );\n      const blobUrl = URL.createObjectURL(imageResponse.data);\n      this.welcomeObjectUrl = blobUrl;\n      this.welcomeData.imageUrl = blobUrl;\n      if (this.welcomeDataHandler) {\n        this.welcomeDataHandler(this.welcomeData);\n      }\n    } catch (err: any) {\n      if (this.isTokenExpiredError(err)) {\n        this.resetChatConversation();\n      }\n      // ignore\n    }\n  }\n\n  // Add feedback to a specific agent message in the conversation\n  async addFeedback(\n    messageId: string,\n    feedback: \"good\" | \"bad\",\n    feedback_message?: string\n  ): Promise<void> {\n    if (!messageId) {\n      throw new Error(\"Message ID is required for feedback\");\n    }\n\n    try {\n      // Use message ID in the URL path, not conversation ID\n      const url = `${this.baseUrl}/api/conversations/message/add-feedback/${messageId}`;\n      // Body should only contain feedback and feedback_message (no message_id)\n      const payload: {\n        feedback: \"good\" | \"bad\";\n        feedback_message?: string;\n      } = {\n        feedback,\n      };\n\n      if (feedback_message) {\n        payload.feedback_message = feedback_message;\n      }\n\n      await axios.patch(url, payload, {\n        headers: this.getHeaders(),\n      });\n\n    } catch (error: any) {\n      if (this.isTokenExpiredError(error)) {\n        this.resetChatConversation();\n      }\n      console.error('Feedback API call failed:', {\n        message: error.message,\n        response: error.response?.data,\n        status: error.response?.status,\n      });\n      throw error;\n    }\n  }\n\n  // Helper method to adjust message timestamps relative to conversation start\n  private adjustMessageTimestamps(message: ChatMessage): ChatMessage {\n    if (!this.conversationCreateTime) {\n      return message;\n    }\n\n    return {\n      ...message,\n      start_time: message.start_time - this.conversationCreateTime,\n      end_time: message.end_time - this.conversationCreateTime,\n    };\n  }\n\n  /************ Heartbeat polling ************/\n  /**\n   * Poll in-progress conversation (heartbeat when WebSocket is disabled).\n   * Returns status and messages normalized to ChatMessage[].\n   */\n  async pollInProgressConversation(): Promise<{\n    status: string;\n    pollMessages: InProgressPollMessage[];\n  }> {\n    if (!this.conversationId || !this.conversationCreateTime) {\n      throw new Error(\"Conversation not started\");\n    }\n\n    const response = await axios.get<InProgressPollResponse>(\n      `${this.baseUrl}/api/conversations/in-progress/poll/${this.conversationId}`,\n      { headers: this.getHeaders() }\n    );\n\n    const pollMessages = (response.data.messages || []).map((m) =>\n      this.normalizePollMessageToChatMessage(m)\n    );\n\n    return { status: response.data.status, pollMessages };\n  }\n\n  private normalizePollMessageToChatMessage(m: InProgressPollMessage): InProgressPollMessage {\n    let createTime =\n      typeof m.create_time === \"number\"\n        ? m.create_time\n        : m.create_time\n          ? +new Date(m.create_time)\n          : Math.floor(Date.now() / 1000);\n\n    // Normalize to seconds: ISO string gives ms (>1e12); if so convert. Otherwise assume already seconds.\n    if (createTime >= 1e12) {\n      createTime = Math.round(createTime / 1000);\n    }\n    const startTime =\n      this.conversationCreateTime != null\n        ? createTime - this.conversationCreateTime\n        : m.start_time;\n    const endTime =\n      this.conversationCreateTime != null\n        ? startTime + (m.end_time - m.start_time) || 0.01\n        : m.end_time;\n    return {\n      id: m.id,\n      create_time: createTime,\n      start_time: startTime,\n      end_time: endTime,\n      speaker: m.speaker as \"customer\" | \"agent\" | \"special\",\n      text: m.text,\n      type: m.type,\n      feedback: m.feedback,\n    };\n  }\n\n  async notifyTakeoverFromPoll(): Promise<void> {\n    // Poll already adds the \"Supervisor took over\" message from the message list.\n    // Only run the handler so UI state updates; do not call handleTakeover() or we duplicate the message.\n    if (this.takeoverHandler) {\n      this.takeoverHandler();\n    }\n  }\n\n  async notifyFinalizedFromPoll(): Promise<void> {\n    // Poll already adds the \"Conversation Finalized\" message from the message list.\n    // Only update state and run the handler; do not call handleConversationFinalized() or we duplicate the message.\n    if (this.finalizedHandler) {\n      this.finalizedHandler();\n    }\n    this.isFinalized = true;\n    this.saveConversation();\n  }\n\n  /****** End of Heartbeat polling ************/\n}\n"]}
|
package/dist/types/index.d.ts
CHANGED
|
@@ -97,6 +97,7 @@ export interface GenAgentChatProps {
|
|
|
97
97
|
tenant: string | undefined;
|
|
98
98
|
metadata?: Record<string, any>;
|
|
99
99
|
useWs?: boolean;
|
|
100
|
+
usePoll?: boolean;
|
|
100
101
|
onError?: (error: Error) => void;
|
|
101
102
|
onTakeover?: () => void;
|
|
102
103
|
onFinalize?: () => void;
|
|
@@ -135,6 +136,10 @@ export interface GenAgentChatProps {
|
|
|
135
136
|
serverUnavailableMessage?: string;
|
|
136
137
|
serverUnavailableContactUrl?: string;
|
|
137
138
|
serverUnavailableContactLabel?: string;
|
|
139
|
+
inputDisclaimer?: string | React.ReactNode;
|
|
140
|
+
onConfigLoaded?: (props: {
|
|
141
|
+
chatInputMetadata?: Record<string, any>;
|
|
142
|
+
}) => void;
|
|
138
143
|
}
|
|
139
144
|
export type AllowedExtension = 'image/*' | 'image/png' | 'image/jpeg' | 'image/jpg' | 'image/gif' | 'application/pdf' | 'application/msword' | 'application/vnd.openxmlformats-officedocument.wordprocessingml.document';
|
|
140
145
|
export type { Translations } from '../utils/i18n';
|
|
@@ -146,3 +151,20 @@ export interface FileUploadResponse {
|
|
|
146
151
|
file_url: string;
|
|
147
152
|
file_id?: string;
|
|
148
153
|
}
|
|
154
|
+
/** Message shape returned by in-progress poll API (backend uses datetime for create_time) */
|
|
155
|
+
export interface InProgressPollMessage {
|
|
156
|
+
id: string;
|
|
157
|
+
create_time?: string | number;
|
|
158
|
+
start_time: number;
|
|
159
|
+
end_time: number;
|
|
160
|
+
speaker: string;
|
|
161
|
+
text: string;
|
|
162
|
+
type?: string;
|
|
163
|
+
sequence_number?: number;
|
|
164
|
+
feedback?: MessageFeedback[];
|
|
165
|
+
}
|
|
166
|
+
/** Response from GET /api/conversations/in-progress/poll/{conversation_id} */
|
|
167
|
+
export interface InProgressPollResponse {
|
|
168
|
+
status: string;
|
|
169
|
+
messages: InProgressPollMessage[];
|
|
170
|
+
}
|