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
package/dist/hooks/useChat.js
CHANGED
|
@@ -54,46 +54,55 @@ var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) {
|
|
|
54
54
|
}
|
|
55
55
|
return to.concat(ar || Array.prototype.slice.call(from));
|
|
56
56
|
};
|
|
57
|
-
import { useState, useEffect, useCallback, useRef, useMemo } from
|
|
58
|
-
import { ChatService } from
|
|
59
|
-
var DEFAULT_SERVER_UNAVAILABLE_MESSAGE =
|
|
60
|
-
var DEFAULT_SERVER_UNAVAILABLE_CONTACT_LABEL =
|
|
57
|
+
import { useState, useEffect, useCallback, useRef, useMemo } from "react";
|
|
58
|
+
import { ChatService } from "../services/chatService";
|
|
59
|
+
var DEFAULT_SERVER_UNAVAILABLE_MESSAGE = "The service is temporarily unavailable. Please try again later.";
|
|
60
|
+
var DEFAULT_SERVER_UNAVAILABLE_CONTACT_LABEL = "Contact support";
|
|
61
61
|
function isNetworkOrServerError(error) {
|
|
62
62
|
var _a;
|
|
63
63
|
// No response = network error
|
|
64
64
|
if (!error.response) {
|
|
65
65
|
var code = error === null || error === void 0 ? void 0 : error.code;
|
|
66
|
-
if (code ===
|
|
66
|
+
if (code === "ERR_NETWORK" ||
|
|
67
|
+
code === "ECONNREFUSED" ||
|
|
68
|
+
code === "ETIMEDOUT" ||
|
|
69
|
+
code === "ERR_CONNECTION_REFUSED")
|
|
67
70
|
return true;
|
|
68
71
|
return true; // any request error without response is treated as server/network issue
|
|
69
72
|
}
|
|
70
73
|
var status = (_a = error.response) === null || _a === void 0 ? void 0 : _a.status;
|
|
71
|
-
return typeof status ===
|
|
74
|
+
return typeof status === "number" && status >= 500;
|
|
72
75
|
}
|
|
73
76
|
export var useChat = function (_a) {
|
|
74
|
-
var baseUrl = _a.baseUrl, apiKey = _a.apiKey, tenant = _a.tenant, metadata = _a.metadata, _b = _a.useWs, useWs = _b === void 0 ? true : _b, language = _a.language, onError = _a.onError, onTakeover = _a.onTakeover, onFinalize = _a.onFinalize, serverUnavailableMessage = _a.serverUnavailableMessage, serverUnavailableContactUrl = _a.serverUnavailableContactUrl, serverUnavailableContactLabel = _a.serverUnavailableContactLabel;
|
|
75
|
-
var
|
|
76
|
-
var
|
|
77
|
-
var
|
|
78
|
-
var
|
|
79
|
-
var
|
|
77
|
+
var baseUrl = _a.baseUrl, apiKey = _a.apiKey, tenant = _a.tenant, metadata = _a.metadata, _b = _a.useWs, useWs = _b === void 0 ? true : _b, _c = _a.usePoll, usePoll = _c === void 0 ? false : _c, language = _a.language, onError = _a.onError, onTakeover = _a.onTakeover, onFinalize = _a.onFinalize, serverUnavailableMessage = _a.serverUnavailableMessage, serverUnavailableContactUrl = _a.serverUnavailableContactUrl, serverUnavailableContactLabel = _a.serverUnavailableContactLabel, onConfigLoaded = _a.onConfigLoaded;
|
|
78
|
+
var _d = useState([]), messages = _d[0], setMessages = _d[1];
|
|
79
|
+
var _e = useState(false), isLoading = _e[0], setIsLoading = _e[1];
|
|
80
|
+
var _f = useState([]), preloadedAttachments = _f[0], setPreloadedAttachments = _f[1];
|
|
81
|
+
var _g = useState("disconnected"), connectionState = _g[0], setConnectionState = _g[1];
|
|
82
|
+
var _h = useState(false), isAgentTyping = _h[0], setIsAgentTyping = _h[1];
|
|
80
83
|
var chatServiceRef = useRef(null);
|
|
81
|
-
var
|
|
82
|
-
var
|
|
83
|
-
var
|
|
84
|
-
var
|
|
85
|
-
var
|
|
86
|
-
var
|
|
87
|
-
var
|
|
88
|
-
var
|
|
89
|
-
var
|
|
90
|
-
var
|
|
84
|
+
var _j = useState(null), conversationId = _j[0], setConversationId = _j[1];
|
|
85
|
+
var _k = useState([]), possibleQueries = _k[0], setPossibleQueries = _k[1];
|
|
86
|
+
var _l = useState(null), welcomeTitle = _l[0], setWelcomeTitle = _l[1];
|
|
87
|
+
var _m = useState(null), welcomeImageUrl = _m[0], setWelcomeImageUrl = _m[1];
|
|
88
|
+
var _o = useState(null), welcomeMessage = _o[0], setWelcomeMessage = _o[1];
|
|
89
|
+
var _p = useState([]), thinkingPhrases = _p[0], setThinkingPhrases = _p[1];
|
|
90
|
+
var _q = useState(1000), thinkingDelayMs = _q[0], setThinkingDelayMs = _q[1];
|
|
91
|
+
var _r = useState({}), chatInputMetadata = _r[0], setChatInputMetadata = _r[1];
|
|
92
|
+
var _s = useState(false), isTakenOver = _s[0], setIsTakenOver = _s[1];
|
|
93
|
+
var _t = useState(false), isFinalized = _t[0], setIsFinalized = _t[1];
|
|
91
94
|
// Scoped messages key for apiKey, conversatioId
|
|
92
95
|
var buildMessagesKey = useCallback(function (apiKeyVal, convId) {
|
|
93
96
|
if (!convId)
|
|
94
97
|
return null;
|
|
95
98
|
return "genassist_conversation_messages:".concat(apiKeyVal, ":").concat(convId);
|
|
96
99
|
}, []);
|
|
100
|
+
var validateMessages = useCallback(function (messages) {
|
|
101
|
+
var hasTakeOverMsg = messages.some(function (m) { return (m === null || m === void 0 ? void 0 : m.type) === "takeover"; });
|
|
102
|
+
var hasFinalizedMsg = messages.some(function (m) { return (m === null || m === void 0 ? void 0 : m.type) === "finalized"; });
|
|
103
|
+
setIsTakenOver(hasTakeOverMsg);
|
|
104
|
+
setIsFinalized(hasFinalizedMsg);
|
|
105
|
+
}, [setIsTakenOver, setIsFinalized]);
|
|
97
106
|
// Check if an error is a token expiration error (401 + "Token has expired.")
|
|
98
107
|
var isTokenExpiredError = useCallback(function (error) {
|
|
99
108
|
var _a, _b;
|
|
@@ -101,14 +110,15 @@ export var useChat = function (_a) {
|
|
|
101
110
|
((_b = error === null || error === void 0 ? void 0 : error.response) === null || _b === void 0 ? void 0 : _b.data) &&
|
|
102
111
|
(error.response.data.error === "Token has expired." ||
|
|
103
112
|
error.response.data.message === "Token has expired." ||
|
|
104
|
-
(typeof error.response.data === "string" &&
|
|
113
|
+
(typeof error.response.data === "string" &&
|
|
114
|
+
error.response.data.includes("Token has expired"))));
|
|
105
115
|
}, []);
|
|
106
116
|
// Reset conversation state to initial (e.g. after token expiration)
|
|
107
117
|
var resetToInitialState = useCallback(function () {
|
|
108
118
|
setConversationId(null);
|
|
109
119
|
setIsFinalized(false);
|
|
110
120
|
setIsTakenOver(false);
|
|
111
|
-
setConnectionState(
|
|
121
|
+
setConnectionState("disconnected");
|
|
112
122
|
setWelcomeTitle(null);
|
|
113
123
|
setWelcomeImageUrl(null);
|
|
114
124
|
setWelcomeMessage(null);
|
|
@@ -117,6 +127,7 @@ export var useChat = function (_a) {
|
|
|
117
127
|
setThinkingDelayMs(1000);
|
|
118
128
|
setChatInputMetadata({});
|
|
119
129
|
setMessages([]);
|
|
130
|
+
lastServerCreateTimeRef.current = 0;
|
|
120
131
|
var key = buildMessagesKey(apiKey, conversationId);
|
|
121
132
|
if (key) {
|
|
122
133
|
try {
|
|
@@ -135,10 +146,19 @@ export var useChat = function (_a) {
|
|
|
135
146
|
var prevApiKeyRef = useRef(apiKey);
|
|
136
147
|
var prevTenantRef = useRef(tenant);
|
|
137
148
|
var prevUseWsRef = useRef(useWs);
|
|
149
|
+
var prevUsePollRef = useRef(usePoll);
|
|
150
|
+
// Heartbeat polling state (when WebSocket is disabled)
|
|
151
|
+
var heartbeatFailureCountRef = useRef(0);
|
|
152
|
+
var heartbeatIntervalRef = useRef(0);
|
|
153
|
+
var lastServerCreateTimeRef = useRef(0);
|
|
154
|
+
// Refs so we only run takeover/finalized once per conversation (avoid stale isTakenOver/isFinalized in poll closure)
|
|
155
|
+
var takeoverProcessedRef = useRef(false);
|
|
156
|
+
var finalizedProcessedRef = useRef(false);
|
|
138
157
|
// Store callbacks in refs so they don't trigger re-initialization
|
|
139
158
|
var onErrorRef = useRef(onError);
|
|
140
159
|
var onTakeoverRef = useRef(onTakeover);
|
|
141
160
|
var onFinalizeRef = useRef(onFinalize);
|
|
161
|
+
var onConfigLoadedRef = useRef(onConfigLoaded);
|
|
142
162
|
// Update callback refs when they change
|
|
143
163
|
useEffect(function () {
|
|
144
164
|
onErrorRef.current = onError;
|
|
@@ -149,16 +169,24 @@ export var useChat = function (_a) {
|
|
|
149
169
|
useEffect(function () {
|
|
150
170
|
onFinalizeRef.current = onFinalize;
|
|
151
171
|
}, [onFinalize]);
|
|
172
|
+
useEffect(function () {
|
|
173
|
+
onConfigLoadedRef.current = onConfigLoaded;
|
|
174
|
+
}, [onConfigLoaded]);
|
|
152
175
|
// Initialize chat service - only when baseUrl, apiKey, tenant, useWs, or metadata actually change
|
|
153
176
|
useEffect(function () {
|
|
154
|
-
var _a, _b, _c, _d, _e, _f, _g, _h;
|
|
177
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _j;
|
|
155
178
|
var metadataChanged = metadataRef.current !== metadataString;
|
|
156
179
|
var baseUrlChanged = prevBaseUrlRef.current !== baseUrl;
|
|
157
180
|
var apiKeyChanged = prevApiKeyRef.current !== apiKey;
|
|
158
181
|
var tenantChanged = prevTenantRef.current !== tenant;
|
|
159
182
|
var useWsChanged = prevUseWsRef.current !== useWs;
|
|
183
|
+
var usePollChanged = prevUsePollRef.current !== usePoll;
|
|
160
184
|
// Only re-initialize for connection-related changes, NOT for metadata changes
|
|
161
|
-
var needsReinit = !chatServiceRef.current ||
|
|
185
|
+
var needsReinit = !chatServiceRef.current ||
|
|
186
|
+
baseUrlChanged ||
|
|
187
|
+
apiKeyChanged ||
|
|
188
|
+
tenantChanged ||
|
|
189
|
+
useWsChanged;
|
|
162
190
|
if (needsReinit) {
|
|
163
191
|
// Update refs
|
|
164
192
|
if (baseUrlChanged)
|
|
@@ -169,6 +197,8 @@ export var useChat = function (_a) {
|
|
|
169
197
|
prevTenantRef.current = tenant;
|
|
170
198
|
if (useWsChanged)
|
|
171
199
|
prevUseWsRef.current = useWs;
|
|
200
|
+
if (usePollChanged)
|
|
201
|
+
prevUsePollRef.current = usePoll;
|
|
172
202
|
if (metadataChanged)
|
|
173
203
|
metadataRef.current = metadataString;
|
|
174
204
|
// Clean up existing service if it exists
|
|
@@ -177,15 +207,30 @@ export var useChat = function (_a) {
|
|
|
177
207
|
chatServiceRef.current.disconnect();
|
|
178
208
|
chatServiceRef.current.setWelcomeDataHandler(null);
|
|
179
209
|
}
|
|
180
|
-
chatServiceRef.current = new ChatService(baseUrl, apiKey, metadata, tenant, language, useWs);
|
|
210
|
+
chatServiceRef.current = new ChatService(baseUrl, apiKey, metadata, tenant, language, useWs, usePoll);
|
|
181
211
|
// Set up handlers
|
|
182
212
|
chatServiceRef.current.setMessageHandler(function (message) {
|
|
183
|
-
var normalizedMessage = __assign(__assign({}, message), { create_time:
|
|
213
|
+
var normalizedMessage = __assign(__assign({}, message), { create_time: !message.create_time || isNaN(message.create_time)
|
|
184
214
|
? Math.floor(Date.now() / 1000)
|
|
185
215
|
: message.create_time });
|
|
186
|
-
|
|
216
|
+
// Track latest create_time we've seen from server/websocket (in seconds)
|
|
217
|
+
if (normalizedMessage.create_time && normalizedMessage.create_time > 0) {
|
|
218
|
+
lastServerCreateTimeRef.current = Math.round(normalizedMessage.create_time) * 1000 / 1000;
|
|
219
|
+
}
|
|
220
|
+
// Add message to messages array
|
|
221
|
+
setMessages(function (prevMessages) {
|
|
222
|
+
// Avoid adding duplicate messages with same message_id and type
|
|
223
|
+
if (normalizedMessage.message_id) {
|
|
224
|
+
var exists = prevMessages.some(function (m) { return m.message_id === normalizedMessage.message_id; });
|
|
225
|
+
if (exists) {
|
|
226
|
+
return prevMessages;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
return __spreadArray(__spreadArray([], prevMessages, true), [normalizedMessage], false);
|
|
230
|
+
});
|
|
187
231
|
// Stop typing animation when agent or system message arrives
|
|
188
|
-
if (normalizedMessage.speaker ===
|
|
232
|
+
if (normalizedMessage.speaker === "agent" ||
|
|
233
|
+
normalizedMessage.speaker === "special") {
|
|
189
234
|
setIsAgentTyping(false);
|
|
190
235
|
}
|
|
191
236
|
});
|
|
@@ -205,7 +250,7 @@ export var useChat = function (_a) {
|
|
|
205
250
|
});
|
|
206
251
|
chatServiceRef.current.setConnectionStateHandler(function (state) {
|
|
207
252
|
setConnectionState(state);
|
|
208
|
-
if (state !==
|
|
253
|
+
if (state !== "connected") {
|
|
209
254
|
setIsAgentTyping(false);
|
|
210
255
|
}
|
|
211
256
|
});
|
|
@@ -229,7 +274,7 @@ export var useChat = function (_a) {
|
|
|
229
274
|
else {
|
|
230
275
|
chatServiceRef.current.connectWebSocket();
|
|
231
276
|
if (!useWs) {
|
|
232
|
-
setConnectionState(
|
|
277
|
+
setConnectionState("connected");
|
|
233
278
|
}
|
|
234
279
|
}
|
|
235
280
|
}
|
|
@@ -250,9 +295,10 @@ export var useChat = function (_a) {
|
|
|
250
295
|
setThinkingDelayMs(thinking.delayMs || 1000);
|
|
251
296
|
}
|
|
252
297
|
var meta = (_h = (_g = chatServiceRef.current).getChatInputMetadata) === null || _h === void 0 ? void 0 : _h.call(_g);
|
|
253
|
-
if (meta && typeof meta ===
|
|
298
|
+
if (meta && typeof meta === "object" && Object.keys(meta).length > 0) {
|
|
254
299
|
setChatInputMetadata(meta);
|
|
255
300
|
}
|
|
301
|
+
(_j = onConfigLoadedRef.current) === null || _j === void 0 ? void 0 : _j.call(onConfigLoadedRef, { chatInputMetadata: meta !== null && meta !== void 0 ? meta : {} });
|
|
256
302
|
}
|
|
257
303
|
}
|
|
258
304
|
else if (chatServiceRef.current) {
|
|
@@ -284,7 +330,18 @@ export var useChat = function (_a) {
|
|
|
284
330
|
return function () {
|
|
285
331
|
// Only cleanup on unmount, not on every dependency change
|
|
286
332
|
};
|
|
287
|
-
}, [
|
|
333
|
+
}, [
|
|
334
|
+
baseUrl,
|
|
335
|
+
apiKey,
|
|
336
|
+
tenant,
|
|
337
|
+
metadataString,
|
|
338
|
+
language,
|
|
339
|
+
useWs,
|
|
340
|
+
usePoll,
|
|
341
|
+
serverUnavailableMessage,
|
|
342
|
+
serverUnavailableContactUrl,
|
|
343
|
+
serverUnavailableContactLabel,
|
|
344
|
+
]);
|
|
288
345
|
// Update language when it changes (without re-initializing the service)
|
|
289
346
|
useEffect(function () {
|
|
290
347
|
if (chatServiceRef.current) {
|
|
@@ -296,16 +353,29 @@ export var useChat = function (_a) {
|
|
|
296
353
|
var key = buildMessagesKey(apiKey, conversationId);
|
|
297
354
|
if (!key) {
|
|
298
355
|
setMessages([]);
|
|
356
|
+
lastServerCreateTimeRef.current = 0;
|
|
299
357
|
return;
|
|
300
358
|
}
|
|
301
359
|
try {
|
|
302
360
|
var stored = localStorage.getItem(key);
|
|
303
|
-
|
|
361
|
+
var parsed = stored ? JSON.parse(stored) : [];
|
|
362
|
+
setMessages(parsed);
|
|
363
|
+
// Initialize last seen create_time from cached messages
|
|
364
|
+
if (parsed.length) {
|
|
365
|
+
var maxCreateTime = Math.max.apply(Math, parsed.map(function (m) { return (m.create_time ? m.create_time : 0); }));
|
|
366
|
+
// Initialize last seen create_time from cached messages
|
|
367
|
+
lastServerCreateTimeRef.current = Math.round(maxCreateTime) * 1000 / 1000;
|
|
368
|
+
}
|
|
304
369
|
}
|
|
305
370
|
catch (error) {
|
|
306
371
|
setMessages([]);
|
|
372
|
+
lastServerCreateTimeRef.current = 0;
|
|
307
373
|
}
|
|
308
374
|
}, [apiKey, conversationId, buildMessagesKey]);
|
|
375
|
+
// Validate messages for takeover and finalized
|
|
376
|
+
useEffect(function () {
|
|
377
|
+
validateMessages(messages);
|
|
378
|
+
}, [messages]);
|
|
309
379
|
// Persist messages for current pair
|
|
310
380
|
useEffect(function () {
|
|
311
381
|
var key = buildMessagesKey(apiKey, conversationId);
|
|
@@ -318,17 +388,169 @@ export var useChat = function (_a) {
|
|
|
318
388
|
// ignore
|
|
319
389
|
}
|
|
320
390
|
}, [messages, apiKey, conversationId, buildMessagesKey]);
|
|
391
|
+
// Heartbeat long polling when WebSocket is disabled and we have an in-progress conversation.
|
|
392
|
+
// - Starts with a short interval
|
|
393
|
+
// - Increases the interval over time (every successful poll)
|
|
394
|
+
// - Retries up to 5 times on errors, then stops
|
|
395
|
+
var HEARTBEAT_INITIAL_INTERVAL_MS = 2000;
|
|
396
|
+
var HEARTBEAT_INTERVAL_STEP_MS = 5000;
|
|
397
|
+
var HEARTBEAT_MAX_INTERVAL_MS = 30000;
|
|
398
|
+
useEffect(function () {
|
|
399
|
+
if (useWs || !conversationId || isFinalized)
|
|
400
|
+
return;
|
|
401
|
+
var svc = chatServiceRef.current;
|
|
402
|
+
if (!svc)
|
|
403
|
+
return;
|
|
404
|
+
// Reset per-conversation flags so we only notify takeover/finalized once
|
|
405
|
+
takeoverProcessedRef.current = false;
|
|
406
|
+
finalizedProcessedRef.current = false;
|
|
407
|
+
var cancelled = false;
|
|
408
|
+
var timeoutId = null;
|
|
409
|
+
var scheduleNext = function (delayMs) {
|
|
410
|
+
if (cancelled)
|
|
411
|
+
return;
|
|
412
|
+
timeoutId = window.setTimeout(poll, delayMs);
|
|
413
|
+
};
|
|
414
|
+
var poll = function () { return __awaiter(void 0, void 0, void 0, function () {
|
|
415
|
+
var _a, status_1, pollMessages, lastServerCreateTime_1, newMessagesRaw, speakerType_1, text_1, newMessages_1, hasTakeoverMessage, hasFinalizedMessage, nextInterval, _b, retryDelay;
|
|
416
|
+
return __generator(this, function (_c) {
|
|
417
|
+
switch (_c.label) {
|
|
418
|
+
case 0:
|
|
419
|
+
if (cancelled)
|
|
420
|
+
return [2 /*return*/];
|
|
421
|
+
_c.label = 1;
|
|
422
|
+
case 1:
|
|
423
|
+
_c.trys.push([1, 3, , 4]);
|
|
424
|
+
return [4 /*yield*/, svc.pollInProgressConversation()];
|
|
425
|
+
case 2:
|
|
426
|
+
_a = _c.sent(), status_1 = _a.status, pollMessages = _a.pollMessages;
|
|
427
|
+
if (cancelled)
|
|
428
|
+
return [2 /*return*/];
|
|
429
|
+
// Reset failure counter on success
|
|
430
|
+
heartbeatFailureCountRef.current = 0;
|
|
431
|
+
lastServerCreateTime_1 = Math.floor(Number(lastServerCreateTimeRef.current) || 0);
|
|
432
|
+
newMessagesRaw = (pollMessages || []).filter(function (m) {
|
|
433
|
+
var ct = Number(m.create_time);
|
|
434
|
+
if (ct === 0 || !Number.isFinite(ct))
|
|
435
|
+
return false;
|
|
436
|
+
return ct > lastServerCreateTime_1;
|
|
437
|
+
});
|
|
438
|
+
if (newMessagesRaw.length > 0) {
|
|
439
|
+
speakerType_1 = function (speaker) {
|
|
440
|
+
if (speaker === "customer")
|
|
441
|
+
return "customer";
|
|
442
|
+
if (speaker === "agent")
|
|
443
|
+
return "agent";
|
|
444
|
+
return "special";
|
|
445
|
+
};
|
|
446
|
+
text_1 = function (m) {
|
|
447
|
+
if (m.type === "takeover")
|
|
448
|
+
return "Supervisor took over";
|
|
449
|
+
if (m.type === "finalized")
|
|
450
|
+
return "Conversation Finalized";
|
|
451
|
+
return m.text;
|
|
452
|
+
};
|
|
453
|
+
newMessages_1 = newMessagesRaw.map(function (m) {
|
|
454
|
+
var _a, _b;
|
|
455
|
+
return ({
|
|
456
|
+
message_id: m.id,
|
|
457
|
+
create_time: Number(m.create_time),
|
|
458
|
+
start_time: Number((_a = m.start_time) !== null && _a !== void 0 ? _a : 0),
|
|
459
|
+
end_time: Number((_b = m.end_time) !== null && _b !== void 0 ? _b : 0),
|
|
460
|
+
speaker: speakerType_1(m.speaker),
|
|
461
|
+
text: text_1(m),
|
|
462
|
+
type: m.type,
|
|
463
|
+
feedback: m.feedback,
|
|
464
|
+
});
|
|
465
|
+
});
|
|
466
|
+
// Append only messages we don't already have (dedupe by message_id; same create_time can repeat every poll)
|
|
467
|
+
if (newMessages_1.length > 0) {
|
|
468
|
+
setMessages(function (prevMessages) {
|
|
469
|
+
var existingIds = new Set(prevMessages.map(function (m) { return m.message_id; }).filter(Boolean));
|
|
470
|
+
var toAdd = newMessages_1.filter(function (m) { return m.message_id == null || !existingIds.has(m.message_id); });
|
|
471
|
+
if (toAdd.length === 0)
|
|
472
|
+
return prevMessages;
|
|
473
|
+
var maxCreateTimeFromPoll = toAdd.reduce(function (max, msg) {
|
|
474
|
+
return Number.isFinite(msg.create_time)
|
|
475
|
+
? Math.max(max, msg.create_time)
|
|
476
|
+
: max;
|
|
477
|
+
}, lastServerCreateTime_1);
|
|
478
|
+
lastServerCreateTimeRef.current = maxCreateTimeFromPoll;
|
|
479
|
+
return __spreadArray(__spreadArray([], prevMessages, true), toAdd, true);
|
|
480
|
+
});
|
|
481
|
+
}
|
|
482
|
+
hasTakeoverMessage = newMessages_1.some(function (m) { return m.type === "takeover"; });
|
|
483
|
+
hasFinalizedMessage = newMessages_1.some(function (m) { return m.type === "finalized"; });
|
|
484
|
+
if ((status_1 === "finalized" || hasFinalizedMessage) && !finalizedProcessedRef.current) {
|
|
485
|
+
finalizedProcessedRef.current = true;
|
|
486
|
+
setIsFinalized(true);
|
|
487
|
+
}
|
|
488
|
+
else if ((status_1 === "takeover" || hasTakeoverMessage) && !takeoverProcessedRef.current) {
|
|
489
|
+
takeoverProcessedRef.current = true;
|
|
490
|
+
setIsTakenOver(true);
|
|
491
|
+
}
|
|
492
|
+
}
|
|
493
|
+
else {
|
|
494
|
+
// No new messages; still honor status if it indicates a terminal state (once per conversation)
|
|
495
|
+
if (status_1 === "finalized" && !finalizedProcessedRef.current) {
|
|
496
|
+
finalizedProcessedRef.current = true;
|
|
497
|
+
svc.handleConversationFinalized();
|
|
498
|
+
setIsFinalized(true);
|
|
499
|
+
}
|
|
500
|
+
else if (status_1 === "takeover" && !takeoverProcessedRef.current) {
|
|
501
|
+
takeoverProcessedRef.current = true;
|
|
502
|
+
svc.notifyTakeoverFromPoll();
|
|
503
|
+
setIsTakenOver(true);
|
|
504
|
+
}
|
|
505
|
+
}
|
|
506
|
+
if (status_1 === "finalized") {
|
|
507
|
+
// Mark finalized and let the effect cleanup prevent further scheduling
|
|
508
|
+
return [2 /*return*/];
|
|
509
|
+
}
|
|
510
|
+
nextInterval = HEARTBEAT_INTERVAL_STEP_MS;
|
|
511
|
+
heartbeatIntervalRef.current = nextInterval;
|
|
512
|
+
scheduleNext(nextInterval);
|
|
513
|
+
return [3 /*break*/, 4];
|
|
514
|
+
case 3:
|
|
515
|
+
_b = _c.sent();
|
|
516
|
+
if (cancelled)
|
|
517
|
+
return [2 /*return*/];
|
|
518
|
+
// Increment failure counter and stop after 5 consecutive failures
|
|
519
|
+
heartbeatFailureCountRef.current += 1;
|
|
520
|
+
if (heartbeatFailureCountRef.current >= 5) {
|
|
521
|
+
return [2 /*return*/];
|
|
522
|
+
}
|
|
523
|
+
retryDelay = heartbeatIntervalRef.current || HEARTBEAT_INITIAL_INTERVAL_MS;
|
|
524
|
+
scheduleNext(retryDelay);
|
|
525
|
+
return [3 /*break*/, 4];
|
|
526
|
+
case 4: return [2 /*return*/];
|
|
527
|
+
}
|
|
528
|
+
});
|
|
529
|
+
}); };
|
|
530
|
+
// Reset counters for this conversation and start immediately
|
|
531
|
+
heartbeatFailureCountRef.current = 0;
|
|
532
|
+
heartbeatIntervalRef.current = HEARTBEAT_INITIAL_INTERVAL_MS;
|
|
533
|
+
if (usePoll && !useWs) {
|
|
534
|
+
poll();
|
|
535
|
+
}
|
|
536
|
+
return function () {
|
|
537
|
+
cancelled = true;
|
|
538
|
+
if (timeoutId !== null) {
|
|
539
|
+
clearTimeout(timeoutId);
|
|
540
|
+
}
|
|
541
|
+
};
|
|
542
|
+
}, [useWs, conversationId, isFinalized, usePoll]);
|
|
321
543
|
// Reset conversation
|
|
322
544
|
var resetConversation = useCallback(function (reCaptchaToken) { return __awaiter(void 0, void 0, void 0, function () {
|
|
323
545
|
var key, convId, queries, welcome, thinking, meta, error_1;
|
|
324
|
-
var _a, _b;
|
|
325
|
-
return __generator(this, function (
|
|
326
|
-
switch (
|
|
546
|
+
var _a, _b, _c, _d, _e, _f;
|
|
547
|
+
return __generator(this, function (_g) {
|
|
548
|
+
switch (_g.label) {
|
|
327
549
|
case 0:
|
|
328
550
|
if (!chatServiceRef.current) {
|
|
329
551
|
return [2 /*return*/];
|
|
330
552
|
}
|
|
331
|
-
setConnectionState(
|
|
553
|
+
setConnectionState("connecting");
|
|
332
554
|
setIsLoading(true);
|
|
333
555
|
setMessages([]);
|
|
334
556
|
setPossibleQueries([]);
|
|
@@ -337,6 +559,7 @@ export var useChat = function (_a) {
|
|
|
337
559
|
setWelcomeMessage(null);
|
|
338
560
|
setThinkingPhrases([]);
|
|
339
561
|
setThinkingDelayMs(1000);
|
|
562
|
+
lastServerCreateTimeRef.current = 0;
|
|
340
563
|
key = buildMessagesKey(apiKey, conversationId);
|
|
341
564
|
if (key) {
|
|
342
565
|
localStorage.removeItem(key);
|
|
@@ -344,16 +567,16 @@ export var useChat = function (_a) {
|
|
|
344
567
|
setIsFinalized(false);
|
|
345
568
|
setIsTakenOver(false);
|
|
346
569
|
setIsAgentTyping(false);
|
|
347
|
-
|
|
570
|
+
_g.label = 1;
|
|
348
571
|
case 1:
|
|
349
|
-
|
|
572
|
+
_g.trys.push([1, 3, 4, 5]);
|
|
350
573
|
// Reset the conversation in the chat service
|
|
351
574
|
chatServiceRef.current.resetChatConversation();
|
|
352
575
|
return [4 /*yield*/, chatServiceRef.current.startConversation(reCaptchaToken)];
|
|
353
576
|
case 2:
|
|
354
|
-
convId =
|
|
577
|
+
convId = _g.sent();
|
|
355
578
|
setConversationId(convId);
|
|
356
|
-
setConnectionState(
|
|
579
|
+
setConnectionState("connected");
|
|
357
580
|
// Get possible queries from API response
|
|
358
581
|
if (chatServiceRef.current.getPossibleQueries) {
|
|
359
582
|
queries = chatServiceRef.current.getPossibleQueries();
|
|
@@ -374,13 +597,16 @@ export var useChat = function (_a) {
|
|
|
374
597
|
setThinkingDelayMs(thinking.delayMs || 1000);
|
|
375
598
|
}
|
|
376
599
|
meta = (_b = (_a = chatServiceRef.current).getChatInputMetadata) === null || _b === void 0 ? void 0 : _b.call(_a);
|
|
377
|
-
if (meta && typeof meta ===
|
|
600
|
+
if (meta && typeof meta === "object" && Object.keys(meta).length > 0) {
|
|
378
601
|
setChatInputMetadata(meta);
|
|
379
602
|
}
|
|
603
|
+
(_c = onConfigLoadedRef.current) === null || _c === void 0 ? void 0 : _c.call(onConfigLoadedRef, {
|
|
604
|
+
chatInputMetadata: (_f = (_e = (_d = chatServiceRef.current).getChatInputMetadata) === null || _e === void 0 ? void 0 : _e.call(_d)) !== null && _f !== void 0 ? _f : {},
|
|
605
|
+
});
|
|
380
606
|
return [3 /*break*/, 5];
|
|
381
607
|
case 3:
|
|
382
|
-
error_1 =
|
|
383
|
-
setConnectionState(
|
|
608
|
+
error_1 = _g.sent();
|
|
609
|
+
setConnectionState("disconnected");
|
|
384
610
|
setIsAgentTyping(false);
|
|
385
611
|
if (onErrorRef.current && error_1 instanceof Error) {
|
|
386
612
|
onErrorRef.current(error_1);
|
|
@@ -434,14 +660,20 @@ export var useChat = function (_a) {
|
|
|
434
660
|
case 4: return [2 /*return*/];
|
|
435
661
|
}
|
|
436
662
|
});
|
|
437
|
-
}); }, [
|
|
663
|
+
}); }, [
|
|
664
|
+
apiKey,
|
|
665
|
+
conversationId,
|
|
666
|
+
buildMessagesKey,
|
|
667
|
+
isTokenExpiredError,
|
|
668
|
+
resetToInitialState,
|
|
669
|
+
]);
|
|
438
670
|
// Send message
|
|
439
|
-
var sendMessage = useCallback(function (
|
|
671
|
+
var sendMessage = useCallback(function (text_2) {
|
|
440
672
|
var args_1 = [];
|
|
441
673
|
for (var _i = 1; _i < arguments.length; _i++) {
|
|
442
674
|
args_1[_i - 1] = arguments[_i];
|
|
443
675
|
}
|
|
444
|
-
return __awaiter(void 0, __spreadArray([
|
|
676
|
+
return __awaiter(void 0, __spreadArray([text_2], args_1, true), void 0, function (text, files, extraMetadata, reCaptchaToken) {
|
|
445
677
|
var newAttachments, uploadedFiles, error_3, now, createTime, startTime, endTime, specialMessage_1;
|
|
446
678
|
var _a;
|
|
447
679
|
if (files === void 0) { files = []; }
|
|
@@ -449,7 +681,7 @@ export var useChat = function (_a) {
|
|
|
449
681
|
switch (_b.label) {
|
|
450
682
|
case 0:
|
|
451
683
|
if (!chatServiceRef.current) {
|
|
452
|
-
throw new Error(
|
|
684
|
+
throw new Error("Chat service not initialized");
|
|
453
685
|
}
|
|
454
686
|
_b.label = 1;
|
|
455
687
|
case 1:
|
|
@@ -457,7 +689,11 @@ export var useChat = function (_a) {
|
|
|
457
689
|
setIsLoading(true);
|
|
458
690
|
newAttachments = [];
|
|
459
691
|
if (files.length > 0) {
|
|
460
|
-
uploadedFiles = files
|
|
692
|
+
uploadedFiles = files
|
|
693
|
+
.map(function (f) {
|
|
694
|
+
return preloadedAttachments.find(function (pa) { return pa.name === f.name && pa.size === f.size; });
|
|
695
|
+
})
|
|
696
|
+
.filter(Boolean);
|
|
461
697
|
newAttachments.push.apply(newAttachments, uploadedFiles);
|
|
462
698
|
}
|
|
463
699
|
// Start typing immediately when user sends, unless conversation is taken over by a human
|
|
@@ -480,7 +716,7 @@ export var useChat = function (_a) {
|
|
|
480
716
|
createTime = (_a = chatServiceRef.current) === null || _a === void 0 ? void 0 : _a.getConversationCreateTime();
|
|
481
717
|
startTime = createTime != null ? now - createTime : 0;
|
|
482
718
|
endTime = startTime + 0.01;
|
|
483
|
-
specialMessage_1 = __assign({ create_time: now, start_time: startTime, end_time: endTime, speaker:
|
|
719
|
+
specialMessage_1 = __assign({ create_time: now, start_time: startTime, end_time: endTime, speaker: "special", text: serverUnavailableMessage !== null && serverUnavailableMessage !== void 0 ? serverUnavailableMessage : DEFAULT_SERVER_UNAVAILABLE_MESSAGE }, (serverUnavailableContactUrl && {
|
|
484
720
|
linkUrl: serverUnavailableContactUrl,
|
|
485
721
|
linkLabel: serverUnavailableContactLabel !== null && serverUnavailableContactLabel !== void 0 ? serverUnavailableContactLabel : DEFAULT_SERVER_UNAVAILABLE_CONTACT_LABEL,
|
|
486
722
|
}));
|
|
@@ -498,20 +734,28 @@ export var useChat = function (_a) {
|
|
|
498
734
|
}
|
|
499
735
|
});
|
|
500
736
|
});
|
|
501
|
-
}, [
|
|
737
|
+
}, [
|
|
738
|
+
preloadedAttachments,
|
|
739
|
+
isTakenOver,
|
|
740
|
+
isTokenExpiredError,
|
|
741
|
+
resetToInitialState,
|
|
742
|
+
serverUnavailableMessage,
|
|
743
|
+
serverUnavailableContactUrl,
|
|
744
|
+
serverUnavailableContactLabel,
|
|
745
|
+
]);
|
|
502
746
|
var startConversation = useCallback(function (reCaptchaToken) { return __awaiter(void 0, void 0, void 0, function () {
|
|
503
747
|
var key, convId, queries, welcome, thinking, meta, error_4;
|
|
504
|
-
var _a, _b;
|
|
505
|
-
return __generator(this, function (
|
|
506
|
-
switch (
|
|
748
|
+
var _a, _b, _c, _d, _e, _f;
|
|
749
|
+
return __generator(this, function (_g) {
|
|
750
|
+
switch (_g.label) {
|
|
507
751
|
case 0:
|
|
508
752
|
if (!chatServiceRef.current) {
|
|
509
753
|
return [2 /*return*/];
|
|
510
754
|
}
|
|
511
|
-
|
|
755
|
+
_g.label = 1;
|
|
512
756
|
case 1:
|
|
513
|
-
|
|
514
|
-
setConnectionState(
|
|
757
|
+
_g.trys.push([1, 3, 4, 5]);
|
|
758
|
+
setConnectionState("connecting");
|
|
515
759
|
setIsLoading(true);
|
|
516
760
|
// Reset state for new conversation
|
|
517
761
|
setMessages([]);
|
|
@@ -526,9 +770,9 @@ export var useChat = function (_a) {
|
|
|
526
770
|
chatServiceRef.current.resetChatConversation();
|
|
527
771
|
return [4 /*yield*/, chatServiceRef.current.startConversation(reCaptchaToken)];
|
|
528
772
|
case 2:
|
|
529
|
-
convId =
|
|
773
|
+
convId = _g.sent();
|
|
530
774
|
setConversationId(convId);
|
|
531
|
-
setConnectionState(
|
|
775
|
+
setConnectionState("connected");
|
|
532
776
|
if (chatServiceRef.current.getPossibleQueries) {
|
|
533
777
|
queries = chatServiceRef.current.getPossibleQueries();
|
|
534
778
|
if (queries && queries.length > 0) {
|
|
@@ -547,13 +791,16 @@ export var useChat = function (_a) {
|
|
|
547
791
|
setThinkingDelayMs(thinking.delayMs || 1000);
|
|
548
792
|
}
|
|
549
793
|
meta = (_b = (_a = chatServiceRef.current).getChatInputMetadata) === null || _b === void 0 ? void 0 : _b.call(_a);
|
|
550
|
-
if (meta && typeof meta ===
|
|
794
|
+
if (meta && typeof meta === "object" && Object.keys(meta).length > 0) {
|
|
551
795
|
setChatInputMetadata(meta);
|
|
552
796
|
}
|
|
797
|
+
(_c = onConfigLoadedRef.current) === null || _c === void 0 ? void 0 : _c.call(onConfigLoadedRef, {
|
|
798
|
+
chatInputMetadata: (_f = (_e = (_d = chatServiceRef.current).getChatInputMetadata) === null || _e === void 0 ? void 0 : _e.call(_d)) !== null && _f !== void 0 ? _f : {},
|
|
799
|
+
});
|
|
553
800
|
return [3 /*break*/, 5];
|
|
554
801
|
case 3:
|
|
555
|
-
error_4 =
|
|
556
|
-
setConnectionState(
|
|
802
|
+
error_4 = _g.sent();
|
|
803
|
+
setConnectionState("disconnected");
|
|
557
804
|
setIsAgentTyping(false);
|
|
558
805
|
if (onErrorRef.current && error_4 instanceof Error) {
|
|
559
806
|
onErrorRef.current(error_4);
|
|
@@ -576,11 +823,11 @@ export var useChat = function (_a) {
|
|
|
576
823
|
switch (_a.label) {
|
|
577
824
|
case 0:
|
|
578
825
|
if (!chatServiceRef.current) {
|
|
579
|
-
console.error(
|
|
826
|
+
console.error("Cannot send feedback: ChatService not initialized");
|
|
580
827
|
return [2 /*return*/];
|
|
581
828
|
}
|
|
582
829
|
if (!messageId) {
|
|
583
|
-
console.error(
|
|
830
|
+
console.error("Cannot send feedback: messageId is required");
|
|
584
831
|
return [2 /*return*/];
|
|
585
832
|
}
|
|
586
833
|
_a.label = 1;
|
|
@@ -635,4 +882,4 @@ export var useChat = function (_a) {
|
|
|
635
882
|
chatInputMetadata: chatInputMetadata,
|
|
636
883
|
};
|
|
637
884
|
};
|
|
638
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"useChat.js","sourceRoot":"","sources":["../../src/hooks/useChat.ts"],"names":[],"mappings":"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","sourcesContent":["import { useState, useEffect, useCallback, useRef, useMemo } from 'react';\nimport { ChatService } from '../services/chatService';\nimport { ChatMessage, Attachment, MessageFeedback } from '../types';\n\nexport interface UseChatProps {\n  baseUrl: string;\n  apiKey: string;\n  tenant?: string | undefined;\n  metadata?: Record<string, any>;\n  //  If false, the chat will run in HTTP-only mode (no WebSocket connection).\n  useWs?: boolean;\n  language?: string;\n  onError?: (error: Error) => void;\n  onTakeover?: () => void;\n  onFinalize?: () => void;\n  serverUnavailableMessage?: string; // Custom message when server is down\n  serverUnavailableContactUrl?: string; // Optional URL for contact/support\n  serverUnavailableContactLabel?: string; // Label for the contact link\n}\n\nconst DEFAULT_SERVER_UNAVAILABLE_MESSAGE = 'The service is temporarily unavailable. Please try again later.';\nconst DEFAULT_SERVER_UNAVAILABLE_CONTACT_LABEL = 'Contact support';\n\nfunction isNetworkOrServerError(error: any): boolean {\n  // No response = network error\n  if (!error.response) {\n    const code = error?.code;\n    if (code === 'ERR_NETWORK' || code === 'ECONNREFUSED' || code === 'ETIMEDOUT' || code === 'ERR_CONNECTION_REFUSED') return true;\n    return true; // any request error without response is treated as server/network issue\n  }\n  const status = error.response?.status;\n  return typeof status === 'number' && status >= 500;\n}\n\nexport const useChat = ({ baseUrl, apiKey, tenant, metadata, useWs = true, language, onError, onTakeover, onFinalize, serverUnavailableMessage, serverUnavailableContactUrl, serverUnavailableContactLabel }: UseChatProps) => {\n  const [messages, setMessages] = useState<ChatMessage[]>([]);\n  const [isLoading, setIsLoading] = useState<boolean>(false);\n  const [preloadedAttachments, setPreloadedAttachments] = useState<Attachment[]>([]);\n  const [connectionState, setConnectionState] = useState<'connecting' | 'connected' | 'disconnected'>('disconnected');\n  const [isAgentTyping, setIsAgentTyping] = useState<boolean>(false);\n  const chatServiceRef = useRef<ChatService | null>(null);\n  const [conversationId, setConversationId] = useState<string | null>(null);\n  const [possibleQueries, setPossibleQueries] = useState<string[]>([]);\n  const [welcomeTitle, setWelcomeTitle] = useState<string | null>(null);\n  const [welcomeImageUrl, setWelcomeImageUrl] = useState<string | null>(null);\n  const [welcomeMessage, setWelcomeMessage] = useState<string | null>(null);\n  const [thinkingPhrases, setThinkingPhrases] = useState<string[]>([]);\n  const [thinkingDelayMs, setThinkingDelayMs] = useState<number>(1000);\n  const [chatInputMetadata, setChatInputMetadata] = useState<Record<string, unknown>>({});\n  const [isTakenOver, setIsTakenOver] = useState<boolean>(false);\n  const [isFinalized, setIsFinalized] = useState<boolean>(false);\n\n  // Scoped messages key for apiKey, conversatioId\n  const buildMessagesKey = useCallback((apiKeyVal: string, convId: string | null) => {\n    if (!convId) return null;\n    return `genassist_conversation_messages:${apiKeyVal}:${convId}`;\n  }, []);\n\n  // Check if an error is a token expiration error (401 + \"Token has expired.\")\n  const isTokenExpiredError = useCallback((error: any): boolean => {\n    return !!(\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  // Reset conversation state to initial (e.g. after token expiration)\n  const resetToInitialState = useCallback(() => {\n    setConversationId(null);\n    setIsFinalized(false);\n    setIsTakenOver(false);\n    setConnectionState('disconnected');\n    setWelcomeTitle(null);\n    setWelcomeImageUrl(null);\n    setWelcomeMessage(null);\n    setPossibleQueries([]);\n    setThinkingPhrases([]);\n    setThinkingDelayMs(1000);\n    setChatInputMetadata({});\n    setMessages([]);\n    const key = buildMessagesKey(apiKey, conversationId);\n    if (key) {\n      try {\n        localStorage.removeItem(key);\n      } catch (e) {\n        // ignore\n      }\n    }\n  }, [apiKey, conversationId, buildMessagesKey]);\n\n  // Deep compare metadata to prevent unnecessary re-initializations\n  // Only re-initialize if metadata actually changes (by value, not reference)\n  const metadataString = useMemo(() => JSON.stringify(metadata || {}), [metadata]);\n  const metadataRef = useRef<string>(metadataString);\n  const prevBaseUrlRef = useRef<string>(baseUrl);\n  const prevApiKeyRef = useRef<string>(apiKey);\n  const prevTenantRef = useRef<string | undefined>(tenant);\n  const prevUseWsRef = useRef<boolean>(useWs);\n\n  // Store callbacks in refs so they don't trigger re-initialization\n  const onErrorRef = useRef(onError);\n  const onTakeoverRef = useRef(onTakeover);\n  const onFinalizeRef = useRef(onFinalize);\n\n  // Update callback refs when they change\n  useEffect(() => {\n    onErrorRef.current = onError;\n  }, [onError]);\n\n  useEffect(() => {\n    onTakeoverRef.current = onTakeover;\n  }, [onTakeover]);\n\n  useEffect(() => {\n    onFinalizeRef.current = onFinalize;\n  }, [onFinalize]);\n\n  // Initialize chat service - only when baseUrl, apiKey, tenant, useWs, or metadata actually change\n  useEffect(() => {\n    const metadataChanged = metadataRef.current !== metadataString;\n    const baseUrlChanged = prevBaseUrlRef.current !== baseUrl;\n    const apiKeyChanged = prevApiKeyRef.current !== apiKey;\n    const tenantChanged = prevTenantRef.current !== tenant;\n    const useWsChanged = prevUseWsRef.current !== useWs;\n    \n    // Only re-initialize for connection-related changes, NOT for metadata changes\n    const needsReinit = !chatServiceRef.current || baseUrlChanged || apiKeyChanged || tenantChanged || useWsChanged;\n\n    if (needsReinit) {\n      // Update refs\n      if (baseUrlChanged) prevBaseUrlRef.current = baseUrl;\n      if (apiKeyChanged) prevApiKeyRef.current = apiKey;\n      if (tenantChanged) prevTenantRef.current = tenant;\n      if (useWsChanged) prevUseWsRef.current = useWs;\n      if (metadataChanged) metadataRef.current = metadataString;\n\n      // Clean up existing service if it exists\n      if (chatServiceRef.current) {\n        chatServiceRef.current.setConnectionStateHandler(() => {});\n        chatServiceRef.current.disconnect();\n        chatServiceRef.current.setWelcomeDataHandler(null);\n      }\n      \n      chatServiceRef.current = new ChatService(baseUrl, apiKey, metadata, tenant, language, useWs);\n      \n      // Set up handlers\n      chatServiceRef.current.setMessageHandler((message: ChatMessage) => {\n        const normalizedMessage: ChatMessage = {\n          ...message,\n          create_time: (!message.create_time || isNaN(message.create_time))\n            ? Math.floor(Date.now() / 1000)\n            : message.create_time,\n        };\n\n        setMessages(prevMessages => [...prevMessages, normalizedMessage]);\n        // Stop typing animation when agent or system message arrives\n        if (normalizedMessage.speaker === 'agent' || normalizedMessage.speaker === 'special') {\n          setIsAgentTyping(false);\n        }\n      });\n\n      chatServiceRef.current.setTakeoverHandler(() => {\n        setIsTakenOver(true);\n        setIsAgentTyping(false);\n        if (onTakeoverRef.current) {\n          onTakeoverRef.current();\n        }\n      });\n\n      chatServiceRef.current.setFinalizedHandler(() => {\n        setIsFinalized(true);\n        setIsAgentTyping(false);\n        if (onFinalizeRef.current) {\n          onFinalizeRef.current();\n        }\n      });\n\n      chatServiceRef.current.setConnectionStateHandler((state) => {\n        setConnectionState(state);\n        if (state !== 'connected') {\n          setIsAgentTyping(false);\n        }\n      });\n\n      chatServiceRef.current.setWelcomeDataHandler((data) => {\n        setWelcomeTitle(data.title ?? null);\n        setWelcomeImageUrl(data.imageUrl ?? null);\n        setWelcomeMessage(data.message ?? null);\n        if (data.possibleQueries && data.possibleQueries.length > 0) {\n          setPossibleQueries(data.possibleQueries);\n        }\n      });\n\n      chatServiceRef.current.setServerUnavailableConfig(\n        serverUnavailableMessage,\n        serverUnavailableContactUrl,\n        serverUnavailableContactLabel\n      );\n\n      // Check for a saved conversation and connect to it\n      const convId = chatServiceRef.current.getConversationId();\n      if (convId) {\n        setConversationId(convId);\n        if (chatServiceRef.current.isConversationFinalized()) {\n          setIsFinalized(true);\n        } else {\n          chatServiceRef.current.connectWebSocket();\n          if (!useWs) {\n            setConnectionState('connected');\n          }\n        }\n      }\n      // Pull initial static data\n      if (chatServiceRef.current) {\n        const queries = chatServiceRef.current.getPossibleQueries?.() || [];\n        if (queries.length) setPossibleQueries(queries);\n        const welcome = chatServiceRef.current.getWelcomeData?.();\n        if (welcome) {\n          setWelcomeTitle(welcome.title || null);\n          setWelcomeImageUrl(welcome.imageUrl || null);\n          setWelcomeMessage(welcome.message || null);\n        }\n        const thinking = chatServiceRef.current.getThinkingConfig?.();\n        if (thinking) {\n          setThinkingPhrases(thinking.phrases || []);\n          setThinkingDelayMs(thinking.delayMs || 1000);\n        }\n        const meta = chatServiceRef.current.getChatInputMetadata?.();\n        if (meta && typeof meta === 'object' && Object.keys(meta).length > 0) {\n          setChatInputMetadata(meta);\n        }\n      }\n    } else if (chatServiceRef.current) {\n      // Just update metadata without re-initializing\n      if (metadataChanged) {\n        metadataRef.current = metadataString;\n        chatServiceRef.current.setMetadata(metadata);\n      }\n    }\n\n    // Always update handlers when callbacks change (without re-initializing)\n    if (chatServiceRef.current) {\n      chatServiceRef.current.setTakeoverHandler(() => {\n        setIsTakenOver(true);\n        setIsAgentTyping(false);\n        if (onTakeoverRef.current) {\n          onTakeoverRef.current();\n        }\n      });\n\n      chatServiceRef.current.setFinalizedHandler(() => {\n        setIsFinalized(true);\n        setIsAgentTyping(false);\n        if (onFinalizeRef.current) {\n          onFinalizeRef.current();\n        }\n      });\n\n      chatServiceRef.current.setServerUnavailableConfig(\n        serverUnavailableMessage,\n        serverUnavailableContactUrl,\n        serverUnavailableContactLabel\n      );\n    }\n\n    // Cleanup only on unmount\n    return () => {\n      // Only cleanup on unmount, not on every dependency change\n    };\n  }, [baseUrl, apiKey, tenant, metadataString, language, useWs, serverUnavailableMessage, serverUnavailableContactUrl, serverUnavailableContactLabel]);\n\n  // Update language when it changes (without re-initializing the service)\n  useEffect(() => {\n    if (chatServiceRef.current) {\n      chatServiceRef.current.setLanguage(language);\n    }\n  }, [language]);\n\n  // Load messages for current pair when available\n  useEffect(() => {\n    const key = buildMessagesKey(apiKey, conversationId);\n    if (!key) {\n      setMessages([]);\n      return;\n    }\n    try {\n      const stored = localStorage.getItem(key);\n      setMessages(stored ? JSON.parse(stored) : []);\n    } catch (error) {\n      setMessages([]);\n    }\n  }, [apiKey, conversationId, buildMessagesKey]);\n\n  // Persist messages for current pair\n  useEffect(() => {\n    const key = buildMessagesKey(apiKey, conversationId);\n    if (!key) return;\n    try {\n      localStorage.setItem(key, JSON.stringify(messages));\n    } catch (error) {\n      // ignore\n    }\n  }, [messages, apiKey, conversationId, buildMessagesKey]);\n\n  // Reset conversation\n  const resetConversation = useCallback(async (reCaptchaToken?: string | undefined) => {\n    if (!chatServiceRef.current) {\n      return;\n    }\n    \n    setConnectionState('connecting');\n    setIsLoading(true);\n    setMessages([]);\n    setPossibleQueries([]);\n    setWelcomeTitle(null);\n    setWelcomeImageUrl(null);\n    setWelcomeMessage(null);\n    setThinkingPhrases([]);\n    setThinkingDelayMs(1000);\n    const key = buildMessagesKey(apiKey, conversationId);\n    if (key) {\n      localStorage.removeItem(key);\n    }\n    setIsFinalized(false);\n    setIsTakenOver(false);\n    setIsAgentTyping(false);\n    \n    try {\n      // Reset the conversation in the chat service\n      chatServiceRef.current.resetChatConversation();\n      \n      // Start a new conversation\n      const convId = await chatServiceRef.current.startConversation(reCaptchaToken);\n      setConversationId(convId);\n      setConnectionState('connected');\n\n      // Get possible queries from API response\n      if (chatServiceRef.current.getPossibleQueries) {\n        const queries = chatServiceRef.current.getPossibleQueries();\n        if (queries && queries.length > 0) {\n          setPossibleQueries(queries);\n        }\n      }\n      // welcome and thinking data\n      if (chatServiceRef.current.getWelcomeData) {\n        const welcome = chatServiceRef.current.getWelcomeData();\n        setWelcomeTitle(welcome.title || null);\n        setWelcomeImageUrl(welcome.imageUrl || null);\n        setWelcomeMessage(welcome.message || null);\n      }\n      if (chatServiceRef.current.getThinkingConfig) {\n        const thinking = chatServiceRef.current.getThinkingConfig();\n        setThinkingPhrases(thinking.phrases || []);\n        setThinkingDelayMs(thinking.delayMs || 1000);\n      }\n      const meta = chatServiceRef.current.getChatInputMetadata?.();\n      if (meta && typeof meta === 'object' && Object.keys(meta).length > 0) {\n        setChatInputMetadata(meta);\n      }\n    } catch (error) {\n      setConnectionState('disconnected');\n      setIsAgentTyping(false);\n      if (onErrorRef.current && error instanceof Error) {\n        onErrorRef.current(error);\n      } else {\n        // ignore\n      }\n    } finally {\n      setIsLoading(false);\n    }\n  }, []);\n\n  const uploadFile = useCallback(async (file: File): Promise<Attachment | null> => {\n    const conversationId = chatServiceRef.current?.getConversationId(); \n    if (!conversationId) {\n      return null;\n    }\n\n    try {\n      const uploadResult = await chatServiceRef.current?.uploadFile(conversationId, file);\n\n      // construct the file url with the base url\n      const file_url = new URL(uploadResult!.file_url!, baseUrl).href;\n      \n      const attachment: Attachment = {\n        name: file.name,\n        type: file.type,\n        size: file.size,\n        url: file_url,\n        file_id: uploadResult?.file_id,\n      };\n\n      setPreloadedAttachments(prev => [...prev, attachment]);\n      return attachment;\n    } catch (error: any) {\n      if (isTokenExpiredError(error)) {\n        resetToInitialState();\n      }\n      if (onErrorRef.current) {\n        onErrorRef.current(error as Error);\n      }\n      return null;\n    }\n  }, [apiKey, conversationId, buildMessagesKey, isTokenExpiredError, resetToInitialState]);\n\n  // Send message\n  const sendMessage = useCallback(async (text: string, files: File[] = [], extraMetadata?: Record<string, any>, reCaptchaToken?: string) => {\n    if (!chatServiceRef.current) {\n      throw new Error('Chat service not initialized');\n    }\n\n    try {\n      setIsLoading(true);\n\n      const newAttachments: Attachment[] = [];\n      \n      if (files.length > 0) {\n        const uploadedFiles = files.map(f => preloadedAttachments.find(pa => pa.name === f.name && pa.size === f.size)).filter(Boolean) as Attachment[];\n        newAttachments.push(...uploadedFiles);\n      }\n\n      // Start typing immediately when user sends, unless conversation is taken over by a human\n      if (!isTakenOver) {\n        setIsAgentTyping(true);\n      }\n      \n      await chatServiceRef.current.sendMessage(text, newAttachments, extraMetadata, reCaptchaToken);\n\n      setPreloadedAttachments([]);\n\n    } catch (error: any) {\n      setIsAgentTyping(false);\n      if (isTokenExpiredError(error)) {\n        resetToInitialState();\n      } else if (isNetworkOrServerError(error)) {\n        // Show custom server-unavailable message\n        const now = Date.now() / 1000;\n        const createTime = chatServiceRef.current?.getConversationCreateTime();\n        const startTime = createTime != null ? now - createTime : 0;\n        const endTime = startTime + 0.01;\n        const specialMessage: ChatMessage = {\n          create_time: now,\n          start_time: startTime,\n          end_time: endTime,\n          speaker: 'special',\n          text: serverUnavailableMessage ?? DEFAULT_SERVER_UNAVAILABLE_MESSAGE,\n          ...(serverUnavailableContactUrl && {\n            linkUrl: serverUnavailableContactUrl,\n            linkLabel: serverUnavailableContactLabel ?? DEFAULT_SERVER_UNAVAILABLE_CONTACT_LABEL,\n          }),\n        };\n        setMessages(prev => [...prev, specialMessage]);\n        // Don't call onError so the user only sees our custom message\n      } else if (onErrorRef.current && error instanceof Error) {\n        onErrorRef.current(error);\n      }\n    } finally {\n      setIsLoading(false);\n    }\n  }, [preloadedAttachments, isTakenOver, isTokenExpiredError, resetToInitialState, serverUnavailableMessage, serverUnavailableContactUrl, serverUnavailableContactLabel]);\n\n  const startConversation = useCallback(async (reCaptchaToken?: string | undefined) => {\n    if (!chatServiceRef.current) {\n      return;\n    }\n    try {\n      setConnectionState('connecting');\n      setIsLoading(true);\n      \n      // Reset state for new conversation\n      setMessages([]);\n      setPossibleQueries([]);\n      const key = buildMessagesKey(apiKey, conversationId);\n      if (key) {\n        localStorage.removeItem(key);\n      }\n      setIsFinalized(false);\n      setIsTakenOver(false);\n      setIsAgentTyping(false);\n      chatServiceRef.current.resetChatConversation();\n\n      const convId = await chatServiceRef.current.startConversation(reCaptchaToken);\n      setConversationId(convId);\n      setConnectionState('connected');\n\n      if (chatServiceRef.current.getPossibleQueries) {\n        const queries = chatServiceRef.current.getPossibleQueries();\n        if (queries && queries.length > 0) {\n          setPossibleQueries(queries);\n        }\n      }\n      if (chatServiceRef.current.getWelcomeData) {\n        const welcome = chatServiceRef.current.getWelcomeData();\n        setWelcomeTitle(welcome.title || null);\n        setWelcomeImageUrl(welcome.imageUrl || null);\n        setWelcomeMessage(welcome.message || null);\n      }\n      if (chatServiceRef.current.getThinkingConfig) {\n        const thinking = chatServiceRef.current.getThinkingConfig();\n        setThinkingPhrases(thinking.phrases || []);\n        setThinkingDelayMs(thinking.delayMs || 1000);\n      }\n      const meta = chatServiceRef.current.getChatInputMetadata?.();\n      if (meta && typeof meta === 'object' && Object.keys(meta).length > 0) {\n        setChatInputMetadata(meta);\n      }\n    } catch (error) {\n      setConnectionState('disconnected');\n      setIsAgentTyping(false);\n      if (onErrorRef.current && error instanceof Error) {\n        onErrorRef.current(error);\n      } else {\n        // ignore\n      }\n    } finally {\n      setIsLoading(false);\n    }\n  }, []);\n\n  // Add feedback to an agent message\n  const addFeedback = useCallback(\n    async (messageId: string, value: 'good' | 'bad', feedbackMessage?: string) => {\n      if (!chatServiceRef.current) {\n        console.error('Cannot send feedback: ChatService not initialized');\n        return;\n      }\n      if (!messageId) {\n        console.error('Cannot send feedback: messageId is required');\n        return;\n      }\n      \n      try {\n        await chatServiceRef.current.addFeedback(messageId, value, feedbackMessage);\n\n        const newFeedback: MessageFeedback = {\n          feedback: value,\n          feedback_message: feedbackMessage,\n          feedback_timestamp: new Date().toISOString(),\n        };\n\n        setMessages(prev =>\n          prev.map(m => {\n            // Match by message_id or id field\n            const msgId = m.message_id || (m as any).id;\n            return msgId === messageId\n              ? { ...m, feedback: [...(m.feedback || []), newFeedback] }\n              : m;\n          })\n        );\n      } catch (error) {\n        if (onErrorRef.current) {\n          onErrorRef.current(error as Error);\n        }\n      }\n    },\n    []\n  );\n\n  return {\n    messages,\n    isLoading,\n    sendMessage,\n    uploadFile,\n    resetConversation,\n    startConversation,\n    connectionState,\n    conversationId,\n    possibleQueries,\n    isTakenOver,\n    isFinalized,\n    isAgentTyping,\n    addFeedback,\n    welcomeTitle,\n    welcomeImageUrl,\n    welcomeMessage,\n    thinkingPhrases,\n    thinkingDelayMs,\n    chatInputMetadata,\n  };\n};\n"]}
|
|
885
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"useChat.js","sourceRoot":"","sources":["../../src/hooks/useChat.ts"],"names":[],"mappings":"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","sourcesContent":["import { useState, useEffect, useCallback, useRef, useMemo } from \"react\";\nimport { ChatService } from \"../services/chatService\";\nimport {\n  ChatMessage,\n  Attachment,\n  MessageFeedback,\n  InProgressPollMessage,\n} from \"../types\";\n\nexport interface UseChatProps {\n  baseUrl: string;\n  apiKey: string;\n  tenant?: string | undefined;\n  metadata?: Record<string, any>;\n  //  If false, the chat will run in HTTP-only mode (no WebSocket connection).\n  useWs?: boolean;\n  usePoll?: boolean;\n  language?: string;\n  onError?: (error: Error) => void;\n  onTakeover?: () => void;\n  onFinalize?: () => void;\n  serverUnavailableMessage?: string; // Custom message when server is down\n  serverUnavailableContactUrl?: string; // Optional URL for contact/support\n  serverUnavailableContactLabel?: string; // Label for the contact link\n  onConfigLoaded?: (props: { chatInputMetadata?: Record<string, any> }) => void; // Callback for when the chat input metadata is loaded\n}\n\nconst DEFAULT_SERVER_UNAVAILABLE_MESSAGE =\n  \"The service is temporarily unavailable. Please try again later.\";\nconst DEFAULT_SERVER_UNAVAILABLE_CONTACT_LABEL = \"Contact support\";\n\nfunction isNetworkOrServerError(error: any): boolean {\n  // No response = network error\n  if (!error.response) {\n    const code = error?.code;\n    if (\n      code === \"ERR_NETWORK\" ||\n      code === \"ECONNREFUSED\" ||\n      code === \"ETIMEDOUT\" ||\n      code === \"ERR_CONNECTION_REFUSED\"\n    )\n      return true;\n    return true; // any request error without response is treated as server/network issue\n  }\n  const status = error.response?.status;\n  return typeof status === \"number\" && status >= 500;\n}\n\nexport const useChat = ({\n  baseUrl,\n  apiKey,\n  tenant,\n  metadata,\n  useWs = true,\n  usePoll = false,\n  language,\n  onError,\n  onTakeover,\n  onFinalize,\n  serverUnavailableMessage,\n  serverUnavailableContactUrl,\n  serverUnavailableContactLabel,\n  onConfigLoaded,\n}: UseChatProps) => {\n  const [messages, setMessages] = useState<ChatMessage[]>([]);\n  const [isLoading, setIsLoading] = useState<boolean>(false);\n  const [preloadedAttachments, setPreloadedAttachments] = useState<\n    Attachment[]\n  >([]);\n  const [connectionState, setConnectionState] = useState<\n    \"connecting\" | \"connected\" | \"disconnected\"\n  >(\"disconnected\");\n  const [isAgentTyping, setIsAgentTyping] = useState<boolean>(false);\n  const chatServiceRef = useRef<ChatService | null>(null);\n  const [conversationId, setConversationId] = useState<string | null>(null);\n  const [possibleQueries, setPossibleQueries] = useState<string[]>([]);\n  const [welcomeTitle, setWelcomeTitle] = useState<string | null>(null);\n  const [welcomeImageUrl, setWelcomeImageUrl] = useState<string | null>(null);\n  const [welcomeMessage, setWelcomeMessage] = useState<string | null>(null);\n  const [thinkingPhrases, setThinkingPhrases] = useState<string[]>([]);\n  const [thinkingDelayMs, setThinkingDelayMs] = useState<number>(1000);\n  const [chatInputMetadata, setChatInputMetadata] = useState<\n    Record<string, unknown>\n  >({});\n  const [isTakenOver, setIsTakenOver] = useState<boolean>(false);\n  const [isFinalized, setIsFinalized] = useState<boolean>(false);\n\n  // Scoped messages key for apiKey, conversatioId\n  const buildMessagesKey = useCallback(\n    (apiKeyVal: string, convId: string | null) => {\n      if (!convId) return null;\n      return `genassist_conversation_messages:${apiKeyVal}:${convId}`;\n    },\n    [],\n  );\n\n  const validateMessages = useCallback(\n    (messages: ChatMessage[]) => {\n      const hasTakeOverMsg = messages.some((m) => m?.type === \"takeover\");\n      const hasFinalizedMsg = messages.some((m) => m?.type === \"finalized\");\n      setIsTakenOver(hasTakeOverMsg);\n      setIsFinalized(hasFinalizedMsg);\n    },\n    [setIsTakenOver, setIsFinalized],\n  );\n\n  // Check if an error is a token expiration error (401 + \"Token has expired.\")\n  const isTokenExpiredError = useCallback((error: any): boolean => {\n    return !!(\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\" &&\n          error.response.data.includes(\"Token has expired\")))\n    );\n  }, []);\n\n  // Reset conversation state to initial (e.g. after token expiration)\n  const resetToInitialState = useCallback(() => {\n    setConversationId(null);\n    setIsFinalized(false);\n    setIsTakenOver(false);\n    setConnectionState(\"disconnected\");\n    setWelcomeTitle(null);\n    setWelcomeImageUrl(null);\n    setWelcomeMessage(null);\n    setPossibleQueries([]);\n    setThinkingPhrases([]);\n    setThinkingDelayMs(1000);\n    setChatInputMetadata({});\n    setMessages([]);\n    lastServerCreateTimeRef.current = 0;\n    const key = buildMessagesKey(apiKey, conversationId);\n    if (key) {\n      try {\n        localStorage.removeItem(key);\n      } catch (e) {\n        // ignore\n      }\n    }\n  }, [apiKey, conversationId, buildMessagesKey]);\n\n  // Deep compare metadata to prevent unnecessary re-initializations\n  // Only re-initialize if metadata actually changes (by value, not reference)\n  const metadataString = useMemo(\n    () => JSON.stringify(metadata || {}),\n    [metadata],\n  );\n  const metadataRef = useRef<string>(metadataString);\n  const prevBaseUrlRef = useRef<string>(baseUrl);\n  const prevApiKeyRef = useRef<string>(apiKey);\n  const prevTenantRef = useRef<string | undefined>(tenant);\n  const prevUseWsRef = useRef<boolean>(useWs);\n  const prevUsePollRef = useRef<boolean>(usePoll);\n\n  // Heartbeat polling state (when WebSocket is disabled)\n  const heartbeatFailureCountRef = useRef<number>(0);\n  const heartbeatIntervalRef = useRef<number>(0);\n  const lastServerCreateTimeRef = useRef<number>(0);\n  // Refs so we only run takeover/finalized once per conversation (avoid stale isTakenOver/isFinalized in poll closure)\n  const takeoverProcessedRef = useRef<boolean>(false);\n  const finalizedProcessedRef = useRef<boolean>(false);\n\n  // Store callbacks in refs so they don't trigger re-initialization\n  const onErrorRef = useRef(onError);\n  const onTakeoverRef = useRef(onTakeover);\n  const onFinalizeRef = useRef(onFinalize);\n  const onConfigLoadedRef = useRef(onConfigLoaded);\n\n  // Update callback refs when they change\n  useEffect(() => {\n    onErrorRef.current = onError;\n  }, [onError]);\n\n  useEffect(() => {\n    onTakeoverRef.current = onTakeover;\n  }, [onTakeover]);\n\n  useEffect(() => {\n    onFinalizeRef.current = onFinalize;\n  }, [onFinalize]);\n\n  useEffect(() => {\n    onConfigLoadedRef.current = onConfigLoaded;\n  }, [onConfigLoaded]);\n\n  // Initialize chat service - only when baseUrl, apiKey, tenant, useWs, or metadata actually change\n  useEffect(() => {\n    const metadataChanged = metadataRef.current !== metadataString;\n    const baseUrlChanged = prevBaseUrlRef.current !== baseUrl;\n    const apiKeyChanged = prevApiKeyRef.current !== apiKey;\n    const tenantChanged = prevTenantRef.current !== tenant;\n    const useWsChanged = prevUseWsRef.current !== useWs;\n    const usePollChanged = prevUsePollRef.current !== usePoll;\n    // Only re-initialize for connection-related changes, NOT for metadata changes\n    const needsReinit =\n      !chatServiceRef.current ||\n      baseUrlChanged ||\n      apiKeyChanged ||\n      tenantChanged ||\n      useWsChanged;\n\n    if (needsReinit) {\n      // Update refs\n      if (baseUrlChanged) prevBaseUrlRef.current = baseUrl;\n      if (apiKeyChanged) prevApiKeyRef.current = apiKey;\n      if (tenantChanged) prevTenantRef.current = tenant;\n      if (useWsChanged) prevUseWsRef.current = useWs;\n      if (usePollChanged) prevUsePollRef.current = usePoll;\n      if (metadataChanged) metadataRef.current = metadataString;\n\n      // Clean up existing service if it exists\n      if (chatServiceRef.current) {\n        chatServiceRef.current.setConnectionStateHandler(() => {});\n        chatServiceRef.current.disconnect();\n        chatServiceRef.current.setWelcomeDataHandler(null);\n      }\n\n      chatServiceRef.current = new ChatService(\n        baseUrl,\n        apiKey,\n        metadata,\n        tenant,\n        language,\n        useWs,\n        usePoll,\n      );\n\n      // Set up handlers\n      chatServiceRef.current.setMessageHandler((message: ChatMessage) => {\n        const normalizedMessage: ChatMessage = {\n          ...message,\n          create_time:\n            !message.create_time || isNaN(message.create_time)\n              ? Math.floor(Date.now() / 1000)\n              : message.create_time,\n        };\n\n        // Track latest create_time we've seen from server/websocket (in seconds)\n        if (normalizedMessage.create_time && normalizedMessage.create_time > 0) {\n          lastServerCreateTimeRef.current = Math.round(normalizedMessage.create_time) * 1000 / 1000;\n        }\n\n        // Add message to messages array\n        setMessages((prevMessages) => {\n          // Avoid adding duplicate messages with same message_id and type\n          if (normalizedMessage.message_id) {\n            const exists = prevMessages.some((m) => m.message_id === normalizedMessage.message_id);\n            if (exists) {\n              return prevMessages;\n            }\n          }\n          return [...prevMessages, normalizedMessage];\n        });\n\n        // Stop typing animation when agent or system message arrives\n        if (\n          normalizedMessage.speaker === \"agent\" ||\n          normalizedMessage.speaker === \"special\"\n        ) {\n          setIsAgentTyping(false);\n        }\n      });\n\n      chatServiceRef.current.setTakeoverHandler(() => {\n        setIsTakenOver(true);\n        setIsAgentTyping(false);\n        if (onTakeoverRef.current) {\n          onTakeoverRef.current();\n        }\n      });\n\n      chatServiceRef.current.setFinalizedHandler(() => {\n        setIsFinalized(true);\n        setIsAgentTyping(false);\n        if (onFinalizeRef.current) {\n          onFinalizeRef.current();\n        }\n      });\n\n      chatServiceRef.current.setConnectionStateHandler((state) => {\n        setConnectionState(state);\n        if (state !== \"connected\") {\n          setIsAgentTyping(false);\n        }\n      });\n\n      chatServiceRef.current.setWelcomeDataHandler((data) => {\n        setWelcomeTitle(data.title ?? null);\n        setWelcomeImageUrl(data.imageUrl ?? null);\n        setWelcomeMessage(data.message ?? null);\n        if (data.possibleQueries && data.possibleQueries.length > 0) {\n          setPossibleQueries(data.possibleQueries);\n        }\n      });\n\n      chatServiceRef.current.setServerUnavailableConfig(\n        serverUnavailableMessage,\n        serverUnavailableContactUrl,\n        serverUnavailableContactLabel,\n      );\n\n      // Check for a saved conversation and connect to it\n      const convId = chatServiceRef.current.getConversationId();\n      if (convId) {\n        setConversationId(convId);\n        if (chatServiceRef.current.isConversationFinalized()) {\n          setIsFinalized(true);\n        } else {\n          chatServiceRef.current.connectWebSocket();\n          if (!useWs) {\n            setConnectionState(\"connected\");\n          }\n        }\n      }\n      // Pull initial static data\n      if (chatServiceRef.current) {\n        const queries = chatServiceRef.current.getPossibleQueries?.() || [];\n        if (queries.length) setPossibleQueries(queries);\n        const welcome = chatServiceRef.current.getWelcomeData?.();\n        if (welcome) {\n          setWelcomeTitle(welcome.title || null);\n          setWelcomeImageUrl(welcome.imageUrl || null);\n          setWelcomeMessage(welcome.message || null);\n        }\n        const thinking = chatServiceRef.current.getThinkingConfig?.();\n        if (thinking) {\n          setThinkingPhrases(thinking.phrases || []);\n          setThinkingDelayMs(thinking.delayMs || 1000);\n        }\n        const meta = chatServiceRef.current.getChatInputMetadata?.();\n        if (meta && typeof meta === \"object\" && Object.keys(meta).length > 0) {\n          setChatInputMetadata(meta);\n        }\n        onConfigLoadedRef.current?.({ chatInputMetadata: meta ?? {} });\n      }\n    } else if (chatServiceRef.current) {\n      // Just update metadata without re-initializing\n      if (metadataChanged) {\n        metadataRef.current = metadataString;\n        chatServiceRef.current.setMetadata(metadata);\n      }\n    }\n\n    // Always update handlers when callbacks change (without re-initializing)\n    if (chatServiceRef.current) {\n      chatServiceRef.current.setTakeoverHandler(() => {\n        setIsTakenOver(true);\n        setIsAgentTyping(false);\n        if (onTakeoverRef.current) {\n          onTakeoverRef.current();\n        }\n      });\n\n      chatServiceRef.current.setFinalizedHandler(() => {\n        setIsFinalized(true);\n        setIsAgentTyping(false);\n        if (onFinalizeRef.current) {\n          onFinalizeRef.current();\n        }\n      });\n\n      chatServiceRef.current.setServerUnavailableConfig(\n        serverUnavailableMessage,\n        serverUnavailableContactUrl,\n        serverUnavailableContactLabel,\n      );\n    }\n\n    // Cleanup only on unmount\n    return () => {\n      // Only cleanup on unmount, not on every dependency change\n    };\n  }, [\n    baseUrl,\n    apiKey,\n    tenant,\n    metadataString,\n    language,\n    useWs,\n    usePoll,\n    serverUnavailableMessage,\n    serverUnavailableContactUrl,\n    serverUnavailableContactLabel,\n  ]);\n\n  // Update language when it changes (without re-initializing the service)\n  useEffect(() => {\n    if (chatServiceRef.current) {\n      chatServiceRef.current.setLanguage(language);\n    }\n  }, [language]);\n\n  // Load messages for current pair when available\n  useEffect(() => {\n    const key = buildMessagesKey(apiKey, conversationId);\n    if (!key) {\n      setMessages([]);\n      lastServerCreateTimeRef.current = 0;\n      return;\n    }\n    try {\n      const stored = localStorage.getItem(key);\n      const parsed: ChatMessage[] = stored ? JSON.parse(stored) : [];\n\n      setMessages(parsed);\n      // Initialize last seen create_time from cached messages\n      if (parsed.length) {\n        const maxCreateTime = Math.max(\n          ...parsed.map((m) => (m.create_time ? m.create_time : 0)),\n        );\n\n        // Initialize last seen create_time from cached messages\n        lastServerCreateTimeRef.current = Math.round(maxCreateTime) * 1000 / 1000;\n      }\n    } catch (error) {\n      setMessages([]);\n      lastServerCreateTimeRef.current = 0;\n    }\n  }, [apiKey, conversationId, buildMessagesKey]);\n\n  // Validate messages for takeover and finalized\n  useEffect(() => {\n    validateMessages(messages);\n  }, [messages]);\n\n  // Persist messages for current pair\n  useEffect(() => {\n    const key = buildMessagesKey(apiKey, conversationId);\n    if (!key) return;\n    try {\n      localStorage.setItem(key, JSON.stringify(messages));\n    } catch (error) {\n      // ignore\n    }\n  }, [messages, apiKey, conversationId, buildMessagesKey]);\n\n  // Heartbeat long polling when WebSocket is disabled and we have an in-progress conversation.\n  // - Starts with a short interval\n  // - Increases the interval over time (every successful poll)\n  // - Retries up to 5 times on errors, then stops\n  const HEARTBEAT_INITIAL_INTERVAL_MS = 2000;\n  const HEARTBEAT_INTERVAL_STEP_MS = 5000;\n  const HEARTBEAT_MAX_INTERVAL_MS = 30000;\n\n  useEffect(() => {\n    if (useWs || !conversationId || isFinalized) return;\n    const svc = chatServiceRef.current;\n    if (!svc) return;\n\n    // Reset per-conversation flags so we only notify takeover/finalized once\n    takeoverProcessedRef.current = false;\n    finalizedProcessedRef.current = false;\n\n    let cancelled = false;\n    let timeoutId: number | null = null;\n\n    const scheduleNext = (delayMs: number) => {\n      if (cancelled) return;\n      timeoutId = window.setTimeout(poll, delayMs);\n    };\n\n    const poll = async () => {\n      if (cancelled) return;\n      try {\n        const { status, pollMessages } = await svc.pollInProgressConversation();\n        if (cancelled) return;\n\n        // Reset failure counter on success\n        heartbeatFailureCountRef.current = 0;\n\n        // Only take messages newer than or equal to last seen create_time (seconds).\n        // Use seconds consistently: poll messages are normalized to seconds in chatService.\n        const lastServerCreateTime = Math.floor(Number(lastServerCreateTimeRef.current) || 0);\n\n        const newMessagesRaw = (pollMessages || []).filter((m) => {\n          const ct = Number((m as any).create_time);\n          if (ct === 0 || !Number.isFinite(ct)) return false;\n          return ct > lastServerCreateTime;\n        });\n\n        if (newMessagesRaw.length > 0) {\n          const speakerType = (speaker: string) => {\n            if (speaker === \"customer\") return \"customer\";\n            if (speaker === \"agent\") return \"agent\";\n            return \"special\";\n          };\n\n          const text = (m: InProgressPollMessage) => {\n            if (m.type === \"takeover\") return \"Supervisor took over\";\n            if (m.type === \"finalized\") return \"Conversation Finalized\";\n            return (m as any).text;\n          };\n\n          const newMessages: ChatMessage[] = newMessagesRaw.map((m) => ({\n            message_id: (m as any).id,\n            create_time: Number((m as any).create_time),\n            start_time: Number((m as any).start_time ?? 0),\n            end_time: Number((m as any).end_time ?? 0),\n            speaker: speakerType((m as any).speaker),\n            text: text(m),\n            type: (m as any).type,\n            feedback: (m as any).feedback,\n          }));\n\n          // Append only messages we don't already have (dedupe by message_id; same create_time can repeat every poll)\n          if (newMessages.length > 0) {\n            setMessages((prevMessages) => {\n              const existingIds = new Set(\n                prevMessages.map((m) => m.message_id).filter(Boolean),\n              );\n              const toAdd = newMessages.filter(\n                (m) => m.message_id == null || !existingIds.has(m.message_id),\n              );\n              if (toAdd.length === 0) return prevMessages;\n              const maxCreateTimeFromPoll = toAdd.reduce(\n                (max, msg) =>\n                  Number.isFinite(msg.create_time)\n                    ? Math.max(max, msg.create_time)\n                    : max,\n                lastServerCreateTime,\n              );\n\n              lastServerCreateTimeRef.current = maxCreateTimeFromPoll;\n              return [...prevMessages, ...toAdd];\n            });\n          }\n\n          // Check new messages for takeover/finalize markers\n          const hasTakeoverMessage = newMessages.some(\n            (m) => m.type === \"takeover\",\n          );\n          const hasFinalizedMessage = newMessages.some(\n            (m) => m.type === \"finalized\",\n          );\n\n          if ((status === \"finalized\" || hasFinalizedMessage) && !finalizedProcessedRef.current) {\n            finalizedProcessedRef.current = true;\n            setIsFinalized(true);\n          } else if ((status === \"takeover\" || hasTakeoverMessage) && !takeoverProcessedRef.current) {\n            takeoverProcessedRef.current = true;\n            setIsTakenOver(true);\n          }\n        } else {\n          // No new messages; still honor status if it indicates a terminal state (once per conversation)\n          if (status === \"finalized\" && !finalizedProcessedRef.current) {\n            finalizedProcessedRef.current = true;\n            svc.handleConversationFinalized();\n            setIsFinalized(true);\n          } else if (status === \"takeover\" && !takeoverProcessedRef.current) {\n            takeoverProcessedRef.current = true;\n            svc.notifyTakeoverFromPoll();\n            setIsTakenOver(true);\n          }\n        }\n\n        if (status === \"finalized\") {\n          // Mark finalized and let the effect cleanup prevent further scheduling\n          return;\n        }\n\n        // Increase polling interval over time on success\n        // const nextInterval = Math.min(\n        //   (heartbeatIntervalRef.current || HEARTBEAT_INITIAL_INTERVAL_MS) +\n        //     HEARTBEAT_INTERVAL_STEP_MS,\n        //   HEARTBEAT_MAX_INTERVAL_MS,\n        // );\n        const nextInterval = HEARTBEAT_INTERVAL_STEP_MS;\n        heartbeatIntervalRef.current = nextInterval;\n        scheduleNext(nextInterval);\n      } catch {\n        if (cancelled) return;\n        // Increment failure counter and stop after 5 consecutive failures\n        heartbeatFailureCountRef.current += 1;\n        if (heartbeatFailureCountRef.current >= 5) {\n          return;\n        }\n        const retryDelay =\n          heartbeatIntervalRef.current || HEARTBEAT_INITIAL_INTERVAL_MS;\n        scheduleNext(retryDelay);\n      }\n    };\n\n    // Reset counters for this conversation and start immediately\n    heartbeatFailureCountRef.current = 0;\n    heartbeatIntervalRef.current = HEARTBEAT_INITIAL_INTERVAL_MS;\n\n    if (usePoll && !useWs) {\n      poll();\n    }\n\n    return () => {\n      cancelled = true;\n      if (timeoutId !== null) {\n        clearTimeout(timeoutId);\n      }\n    };\n  }, [useWs, conversationId, isFinalized, usePoll]);\n\n  // Reset conversation\n  const resetConversation = useCallback(\n    async (reCaptchaToken?: string | undefined) => {\n      if (!chatServiceRef.current) {\n        return;\n      }\n\n      setConnectionState(\"connecting\");\n      setIsLoading(true);\n      setMessages([]);\n      setPossibleQueries([]);\n      setWelcomeTitle(null);\n      setWelcomeImageUrl(null);\n      setWelcomeMessage(null);\n      setThinkingPhrases([]);\n      setThinkingDelayMs(1000);\n      lastServerCreateTimeRef.current = 0;\n      const key = buildMessagesKey(apiKey, conversationId);\n      if (key) {\n        localStorage.removeItem(key);\n      }\n      setIsFinalized(false);\n      setIsTakenOver(false);\n      setIsAgentTyping(false);\n\n      try {\n        // Reset the conversation in the chat service\n        chatServiceRef.current.resetChatConversation();\n\n        // Start a new conversation\n        const convId =\n          await chatServiceRef.current.startConversation(reCaptchaToken);\n        setConversationId(convId);\n        setConnectionState(\"connected\");\n\n        // Get possible queries from API response\n        if (chatServiceRef.current.getPossibleQueries) {\n          const queries = chatServiceRef.current.getPossibleQueries();\n          if (queries && queries.length > 0) {\n            setPossibleQueries(queries);\n          }\n        }\n        // welcome and thinking data\n        if (chatServiceRef.current.getWelcomeData) {\n          const welcome = chatServiceRef.current.getWelcomeData();\n          setWelcomeTitle(welcome.title || null);\n          setWelcomeImageUrl(welcome.imageUrl || null);\n          setWelcomeMessage(welcome.message || null);\n        }\n        if (chatServiceRef.current.getThinkingConfig) {\n          const thinking = chatServiceRef.current.getThinkingConfig();\n          setThinkingPhrases(thinking.phrases || []);\n          setThinkingDelayMs(thinking.delayMs || 1000);\n        }\n        const meta = chatServiceRef.current.getChatInputMetadata?.();\n        if (meta && typeof meta === \"object\" && Object.keys(meta).length > 0) {\n          setChatInputMetadata(meta);\n        }\n        onConfigLoadedRef.current?.({\n          chatInputMetadata:\n            chatServiceRef.current.getChatInputMetadata?.() ?? {},\n        });\n      } catch (error) {\n        setConnectionState(\"disconnected\");\n        setIsAgentTyping(false);\n        if (onErrorRef.current && error instanceof Error) {\n          onErrorRef.current(error);\n        } else {\n          // ignore\n        }\n      } finally {\n        setIsLoading(false);\n      }\n    },\n    [],\n  );\n\n  const uploadFile = useCallback(\n    async (file: File): Promise<Attachment | null> => {\n      const conversationId = chatServiceRef.current?.getConversationId();\n      if (!conversationId) {\n        return null;\n      }\n\n      try {\n        const uploadResult = await chatServiceRef.current?.uploadFile(\n          conversationId,\n          file,\n        );\n\n        // construct the file url with the base url\n        const file_url = new URL(uploadResult!.file_url!, baseUrl).href;\n\n        const attachment: Attachment = {\n          name: file.name,\n          type: file.type,\n          size: file.size,\n          url: file_url,\n          file_id: uploadResult?.file_id,\n        };\n\n        setPreloadedAttachments((prev) => [...prev, attachment]);\n        return attachment;\n      } catch (error: any) {\n        if (isTokenExpiredError(error)) {\n          resetToInitialState();\n        }\n        if (onErrorRef.current) {\n          onErrorRef.current(error as Error);\n        }\n        return null;\n      }\n    },\n    [\n      apiKey,\n      conversationId,\n      buildMessagesKey,\n      isTokenExpiredError,\n      resetToInitialState,\n    ],\n  );\n\n  // Send message\n  const sendMessage = useCallback(\n    async (\n      text: string,\n      files: File[] = [],\n      extraMetadata?: Record<string, any>,\n      reCaptchaToken?: string,\n    ) => {\n      if (!chatServiceRef.current) {\n        throw new Error(\"Chat service not initialized\");\n      }\n\n      try {\n        setIsLoading(true);\n\n        const newAttachments: Attachment[] = [];\n\n        if (files.length > 0) {\n          const uploadedFiles = files\n            .map((f) =>\n              preloadedAttachments.find(\n                (pa) => pa.name === f.name && pa.size === f.size,\n              ),\n            )\n            .filter(Boolean) as Attachment[];\n          newAttachments.push(...uploadedFiles);\n        }\n\n        // Start typing immediately when user sends, unless conversation is taken over by a human\n        if (!isTakenOver) {\n          setIsAgentTyping(true);\n        }\n\n        await chatServiceRef.current.sendMessage(\n          text,\n          newAttachments,\n          extraMetadata,\n          reCaptchaToken,\n        );\n\n        setPreloadedAttachments([]);\n      } catch (error: any) {\n        setIsAgentTyping(false);\n        if (isTokenExpiredError(error)) {\n          resetToInitialState();\n        } else if (isNetworkOrServerError(error)) {\n          // Show custom server-unavailable message\n          const now = Date.now() / 1000;\n          const createTime =\n            chatServiceRef.current?.getConversationCreateTime();\n          const startTime = createTime != null ? now - createTime : 0;\n          const endTime = startTime + 0.01;\n          const specialMessage: ChatMessage = {\n            create_time: now,\n            start_time: startTime,\n            end_time: endTime,\n            speaker: \"special\",\n            text:\n              serverUnavailableMessage ?? DEFAULT_SERVER_UNAVAILABLE_MESSAGE,\n            ...(serverUnavailableContactUrl && {\n              linkUrl: serverUnavailableContactUrl,\n              linkLabel:\n                serverUnavailableContactLabel ??\n                DEFAULT_SERVER_UNAVAILABLE_CONTACT_LABEL,\n            }),\n          };\n\n          setMessages((prev) => [...prev, specialMessage]);\n          // Don't call onError so the user only sees our custom message\n        } else if (onErrorRef.current && error instanceof Error) {\n          onErrorRef.current(error);\n        }\n      } finally {\n        setIsLoading(false);\n      }\n    },\n    [\n      preloadedAttachments,\n      isTakenOver,\n      isTokenExpiredError,\n      resetToInitialState,\n      serverUnavailableMessage,\n      serverUnavailableContactUrl,\n      serverUnavailableContactLabel,\n    ],\n  );\n\n  const startConversation = useCallback(\n    async (reCaptchaToken?: string | undefined) => {\n      if (!chatServiceRef.current) {\n        return;\n      }\n      try {\n        setConnectionState(\"connecting\");\n        setIsLoading(true);\n\n        // Reset state for new conversation\n        setMessages([]);\n        setPossibleQueries([]);\n        const key = buildMessagesKey(apiKey, conversationId);\n        if (key) {\n          localStorage.removeItem(key);\n        }\n        setIsFinalized(false);\n        setIsTakenOver(false);\n        setIsAgentTyping(false);\n        chatServiceRef.current.resetChatConversation();\n\n        const convId =\n          await chatServiceRef.current.startConversation(reCaptchaToken);\n        setConversationId(convId);\n        setConnectionState(\"connected\");\n\n        if (chatServiceRef.current.getPossibleQueries) {\n          const queries = chatServiceRef.current.getPossibleQueries();\n          if (queries && queries.length > 0) {\n            setPossibleQueries(queries);\n          }\n        }\n        if (chatServiceRef.current.getWelcomeData) {\n          const welcome = chatServiceRef.current.getWelcomeData();\n          setWelcomeTitle(welcome.title || null);\n          setWelcomeImageUrl(welcome.imageUrl || null);\n          setWelcomeMessage(welcome.message || null);\n        }\n        if (chatServiceRef.current.getThinkingConfig) {\n          const thinking = chatServiceRef.current.getThinkingConfig();\n          setThinkingPhrases(thinking.phrases || []);\n          setThinkingDelayMs(thinking.delayMs || 1000);\n        }\n        const meta = chatServiceRef.current.getChatInputMetadata?.();\n        if (meta && typeof meta === \"object\" && Object.keys(meta).length > 0) {\n          setChatInputMetadata(meta);\n        }\n        onConfigLoadedRef.current?.({\n          chatInputMetadata:\n            chatServiceRef.current.getChatInputMetadata?.() ?? {},\n        });\n      } catch (error) {\n        setConnectionState(\"disconnected\");\n        setIsAgentTyping(false);\n        if (onErrorRef.current && error instanceof Error) {\n          onErrorRef.current(error);\n        } else {\n          // ignore\n        }\n      } finally {\n        setIsLoading(false);\n      }\n    },\n    [],\n  );\n\n  // Add feedback to an agent message\n  const addFeedback = useCallback(\n    async (\n      messageId: string,\n      value: \"good\" | \"bad\",\n      feedbackMessage?: string,\n    ) => {\n      if (!chatServiceRef.current) {\n        console.error(\"Cannot send feedback: ChatService not initialized\");\n        return;\n      }\n      if (!messageId) {\n        console.error(\"Cannot send feedback: messageId is required\");\n        return;\n      }\n\n      try {\n        await chatServiceRef.current.addFeedback(\n          messageId,\n          value,\n          feedbackMessage,\n        );\n\n        const newFeedback: MessageFeedback = {\n          feedback: value,\n          feedback_message: feedbackMessage,\n          feedback_timestamp: new Date().toISOString(),\n        };\n\n        setMessages((prev) =>\n          prev.map((m) => {\n            // Match by message_id or id field\n            const msgId = m.message_id || (m as any).id;\n            return msgId === messageId\n              ? { ...m, feedback: [...(m.feedback || []), newFeedback] }\n              : m;\n          }),\n        );\n      } catch (error) {\n        if (onErrorRef.current) {\n          onErrorRef.current(error as Error);\n        }\n      }\n    },\n    [],\n  );\n\n  return {\n    messages,\n    isLoading,\n    sendMessage,\n    uploadFile,\n    resetConversation,\n    startConversation,\n    connectionState,\n    conversationId,\n    possibleQueries,\n    isTakenOver,\n    isFinalized,\n    isAgentTyping,\n    addFeedback,\n    welcomeTitle,\n    welcomeImageUrl,\n    welcomeMessage,\n    thinkingPhrases,\n    thinkingDelayMs,\n    chatInputMetadata,\n  };\n};\n"]}
|