genassist-chat-react 1.0.3 → 1.0.4
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/ChatMessage.d.ts +6 -0
- package/dist/components/ChatMessage.js +181 -54
- package/dist/components/GenAgentChat.js +178 -42
- package/dist/components/VoiceInput.js +12 -10
- package/dist/hooks/useChat.d.ts +7 -1
- package/dist/hooks/useChat.js +104 -10
- package/dist/services/chatService.d.ts +10 -3
- package/dist/services/chatService.js +168 -73
- package/dist/types/index.d.ts +27 -3
- package/dist/types/index.js +1 -1
- package/package.json +3 -2
- package/dist/assets/chat-logo.png +0 -0
|
@@ -45,10 +45,10 @@ var __generator = (this && this.__generator) || function (thisArg, body) {
|
|
|
45
45
|
if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
|
|
46
46
|
}
|
|
47
47
|
};
|
|
48
|
-
import axios from
|
|
49
|
-
import { w3cwebsocket as WebSocket } from
|
|
48
|
+
import axios from "axios";
|
|
49
|
+
import { w3cwebsocket as WebSocket, } from "websocket";
|
|
50
50
|
var ChatService = /** @class */ (function () {
|
|
51
|
-
function ChatService(baseUrl, apiKey, metadata) {
|
|
51
|
+
function ChatService(baseUrl, apiKey, metadata, tenant) {
|
|
52
52
|
this.conversationId = null;
|
|
53
53
|
this.conversationCreateTime = null; // Track conversation start time
|
|
54
54
|
this.isFinalized = false;
|
|
@@ -57,11 +57,15 @@ var ChatService = /** @class */ (function () {
|
|
|
57
57
|
this.takeoverHandler = null;
|
|
58
58
|
this.finalizedHandler = null;
|
|
59
59
|
this.connectionStateHandler = null;
|
|
60
|
-
this.storageKey =
|
|
60
|
+
this.storageKey = "genassist_conversation";
|
|
61
61
|
this.possibleQueries = [];
|
|
62
|
-
this.
|
|
62
|
+
this.welcomeData = {};
|
|
63
|
+
this.thinkingConfig = { phrases: [], delayMs: 1000 };
|
|
64
|
+
this.welcomeObjectUrl = null; // to revoke on reset
|
|
65
|
+
this.baseUrl = baseUrl.endsWith("/") ? baseUrl.slice(0, -1) : baseUrl;
|
|
63
66
|
this.apiKey = apiKey;
|
|
64
67
|
this.metadata = metadata;
|
|
68
|
+
this.tenant = tenant;
|
|
65
69
|
// Try to load a saved conversation ID from localStorage
|
|
66
70
|
this.loadSavedConversation();
|
|
67
71
|
}
|
|
@@ -80,6 +84,12 @@ var ChatService = /** @class */ (function () {
|
|
|
80
84
|
ChatService.prototype.getPossibleQueries = function () {
|
|
81
85
|
return this.possibleQueries;
|
|
82
86
|
};
|
|
87
|
+
ChatService.prototype.getWelcomeData = function () {
|
|
88
|
+
return this.welcomeData;
|
|
89
|
+
};
|
|
90
|
+
ChatService.prototype.getThinkingConfig = function () {
|
|
91
|
+
return this.thinkingConfig;
|
|
92
|
+
};
|
|
83
93
|
/**
|
|
84
94
|
* Load a saved conversation ID from localStorage
|
|
85
95
|
*/
|
|
@@ -91,11 +101,11 @@ var ChatService = /** @class */ (function () {
|
|
|
91
101
|
this.conversationId = conversationId;
|
|
92
102
|
this.conversationCreateTime = createTime;
|
|
93
103
|
this.isFinalized = isFinalized || false;
|
|
94
|
-
console.log(
|
|
104
|
+
console.log("Loaded saved conversation:", this.conversationId, "Finalized:", this.isFinalized);
|
|
95
105
|
}
|
|
96
106
|
}
|
|
97
107
|
catch (error) {
|
|
98
|
-
console.error(
|
|
108
|
+
console.error("Error loading saved conversation:", error);
|
|
99
109
|
}
|
|
100
110
|
};
|
|
101
111
|
/**
|
|
@@ -110,11 +120,11 @@ var ChatService = /** @class */ (function () {
|
|
|
110
120
|
isFinalized: this.isFinalized,
|
|
111
121
|
};
|
|
112
122
|
localStorage.setItem(this.storageKey, JSON.stringify(conversationData));
|
|
113
|
-
console.log(
|
|
123
|
+
console.log("Saved conversation:", this.conversationId);
|
|
114
124
|
}
|
|
115
125
|
}
|
|
116
126
|
catch (error) {
|
|
117
|
-
console.error(
|
|
127
|
+
console.error("Error saving conversation:", error);
|
|
118
128
|
}
|
|
119
129
|
};
|
|
120
130
|
/**
|
|
@@ -132,12 +142,21 @@ var ChatService = /** @class */ (function () {
|
|
|
132
142
|
this.isFinalized = false;
|
|
133
143
|
// Clear possible queries
|
|
134
144
|
this.possibleQueries = [];
|
|
145
|
+
this.welcomeData = {};
|
|
146
|
+
this.thinkingConfig = { phrases: [], delayMs: 1000 };
|
|
147
|
+
if (this.welcomeObjectUrl) {
|
|
148
|
+
try {
|
|
149
|
+
URL.revokeObjectURL(this.welcomeObjectUrl);
|
|
150
|
+
}
|
|
151
|
+
catch (_a) { }
|
|
152
|
+
this.welcomeObjectUrl = null;
|
|
153
|
+
}
|
|
135
154
|
// Remove from local storage
|
|
136
155
|
try {
|
|
137
156
|
localStorage.removeItem(this.storageKey);
|
|
138
157
|
}
|
|
139
158
|
catch (error) {
|
|
140
|
-
console.error(
|
|
159
|
+
console.error("Error removing conversation from storage:", error);
|
|
141
160
|
}
|
|
142
161
|
};
|
|
143
162
|
/**
|
|
@@ -157,55 +176,93 @@ var ChatService = /** @class */ (function () {
|
|
|
157
176
|
};
|
|
158
177
|
ChatService.prototype.startConversation = function () {
|
|
159
178
|
return __awaiter(this, void 0, void 0, function () {
|
|
160
|
-
var requestBody, response, now, welcomeMessage, error_1;
|
|
179
|
+
var requestBody, response, anyData, agentId, welcomeTitle, welcomeImageUrl, thinkingPhrases, thinkingDelaySec, imageResponse, blobUrl, err_1, now, welcomeMessage, error_1;
|
|
161
180
|
return __generator(this, function (_a) {
|
|
162
181
|
switch (_a.label) {
|
|
163
182
|
case 0:
|
|
164
|
-
_a.trys.push([0,
|
|
183
|
+
_a.trys.push([0, 6, , 7]);
|
|
165
184
|
requestBody = {
|
|
166
185
|
messages: [],
|
|
167
186
|
recorded_at: new Date().toISOString(),
|
|
168
|
-
data_source_id: "00000000-0000-0000-0000-000000000000"
|
|
187
|
+
data_source_id: "00000000-0000-0000-0000-000000000000",
|
|
169
188
|
};
|
|
170
189
|
if (this.metadata) {
|
|
171
190
|
requestBody.metadata = this.metadata;
|
|
172
191
|
}
|
|
173
192
|
return [4 /*yield*/, axios.post("".concat(this.baseUrl, "/api/conversations/in-progress/start"), requestBody, {
|
|
174
|
-
headers: {
|
|
175
|
-
'x-api-key': this.apiKey,
|
|
176
|
-
'Content-Type': 'application/json'
|
|
177
|
-
}
|
|
193
|
+
headers: __assign({ "x-api-key": this.apiKey, "Content-Type": "application/json" }, (this.tenant ? { "X-Tenant-Id": this.tenant } : {})),
|
|
178
194
|
})];
|
|
179
195
|
case 1:
|
|
180
196
|
response = _a.sent();
|
|
181
197
|
this.conversationId = response.data.conversation_id;
|
|
182
198
|
// Store conversation create time (use from response if available, otherwise current time)
|
|
183
|
-
this.conversationCreateTime = response.data.create_time
|
|
199
|
+
this.conversationCreateTime = response.data.create_time
|
|
200
|
+
? response.data.create_time / 1000
|
|
201
|
+
: Date.now() / 1000;
|
|
184
202
|
this.isFinalized = false;
|
|
185
203
|
this.saveConversation();
|
|
186
204
|
this.connectWebSocket();
|
|
187
205
|
// Store possible queries if available
|
|
188
|
-
if (response.data.agent_possible_queries &&
|
|
206
|
+
if (response.data.agent_possible_queries &&
|
|
207
|
+
response.data.agent_possible_queries.length > 0) {
|
|
189
208
|
this.possibleQueries = response.data.agent_possible_queries;
|
|
190
209
|
}
|
|
210
|
+
anyData = response.data;
|
|
211
|
+
agentId = anyData.agent_id;
|
|
212
|
+
welcomeTitle = anyData.agent_welcome_title;
|
|
213
|
+
welcomeImageUrl = anyData.agent_welcome_image_url;
|
|
214
|
+
thinkingPhrases = anyData.agent_thinking_phrases;
|
|
215
|
+
thinkingDelaySec = anyData.agent_thinking_phrase_delay;
|
|
216
|
+
this.welcomeData = {
|
|
217
|
+
title: welcomeTitle || null,
|
|
218
|
+
message: null,
|
|
219
|
+
imageUrl: welcomeImageUrl || null,
|
|
220
|
+
possibleQueries: this.possibleQueries,
|
|
221
|
+
};
|
|
222
|
+
if (Array.isArray(thinkingPhrases) && thinkingPhrases.length > 0) {
|
|
223
|
+
this.thinkingConfig.phrases = thinkingPhrases;
|
|
224
|
+
}
|
|
225
|
+
if (typeof thinkingDelaySec === "number" && thinkingDelaySec >= 0) {
|
|
226
|
+
this.thinkingConfig.delayMs = Math.max(250, Math.round(thinkingDelaySec * 1000));
|
|
227
|
+
}
|
|
228
|
+
if (!(!this.welcomeData.imageUrl && agentId)) return [3 /*break*/, 5];
|
|
229
|
+
_a.label = 2;
|
|
230
|
+
case 2:
|
|
231
|
+
_a.trys.push([2, 4, , 5]);
|
|
232
|
+
return [4 /*yield*/, axios.get("".concat(this.baseUrl, "/api/genagent/agents/configs/").concat(agentId, "/welcome-image"), {
|
|
233
|
+
headers: __assign({ "x-api-key": this.apiKey }, (this.tenant ? { "X-Tenant-Id": this.tenant } : {})),
|
|
234
|
+
responseType: "blob",
|
|
235
|
+
})];
|
|
236
|
+
case 3:
|
|
237
|
+
imageResponse = _a.sent();
|
|
238
|
+
blobUrl = URL.createObjectURL(imageResponse.data);
|
|
239
|
+
this.welcomeObjectUrl = blobUrl;
|
|
240
|
+
this.welcomeData.imageUrl = blobUrl;
|
|
241
|
+
return [3 /*break*/, 5];
|
|
242
|
+
case 4:
|
|
243
|
+
err_1 = _a.sent();
|
|
244
|
+
console.log("No welcome image available or failed to load.");
|
|
245
|
+
return [3 /*break*/, 5];
|
|
246
|
+
case 5:
|
|
191
247
|
// Process agent welcome message if available
|
|
192
248
|
if (response.data.agent_welcome_message && this.messageHandler) {
|
|
193
249
|
now = Date.now() / 1000;
|
|
194
250
|
welcomeMessage = {
|
|
195
251
|
create_time: now,
|
|
196
252
|
start_time: now - this.conversationCreateTime, // Relative to conversation start
|
|
197
|
-
end_time:
|
|
198
|
-
speaker:
|
|
199
|
-
text: response.data.agent_welcome_message
|
|
253
|
+
end_time: now - this.conversationCreateTime + 0.01, // Relative to conversation start
|
|
254
|
+
speaker: "agent",
|
|
255
|
+
text: response.data.agent_welcome_message,
|
|
200
256
|
};
|
|
257
|
+
this.welcomeData.message = response.data.agent_welcome_message;
|
|
201
258
|
this.messageHandler(welcomeMessage);
|
|
202
259
|
}
|
|
203
260
|
return [2 /*return*/, response.data.conversation_id];
|
|
204
|
-
case
|
|
261
|
+
case 6:
|
|
205
262
|
error_1 = _a.sent();
|
|
206
|
-
console.error(
|
|
263
|
+
console.error("Error starting conversation:", error_1);
|
|
207
264
|
throw error_1;
|
|
208
|
-
case
|
|
265
|
+
case 7: return [2 /*return*/];
|
|
209
266
|
}
|
|
210
267
|
});
|
|
211
268
|
});
|
|
@@ -217,14 +274,14 @@ var ChatService = /** @class */ (function () {
|
|
|
217
274
|
switch (_a.label) {
|
|
218
275
|
case 0:
|
|
219
276
|
if (!this.conversationId || !this.conversationCreateTime) {
|
|
220
|
-
throw new Error(
|
|
277
|
+
throw new Error("Conversation not started");
|
|
221
278
|
}
|
|
222
279
|
now = Date.now() / 1000;
|
|
223
280
|
chatMessage = {
|
|
224
281
|
create_time: now,
|
|
225
282
|
start_time: now - this.conversationCreateTime, // Relative to conversation start
|
|
226
|
-
end_time:
|
|
227
|
-
speaker:
|
|
283
|
+
end_time: now - this.conversationCreateTime + 0.01, // Relative to conversation start
|
|
284
|
+
speaker: "customer",
|
|
228
285
|
text: message,
|
|
229
286
|
attachments: attachments,
|
|
230
287
|
};
|
|
@@ -235,29 +292,28 @@ var ChatService = /** @class */ (function () {
|
|
|
235
292
|
case 1:
|
|
236
293
|
_a.trys.push([1, 3, , 4]);
|
|
237
294
|
return [4 /*yield*/, axios.patch("".concat(this.baseUrl, "/api/conversations/in-progress/update/").concat(this.conversationId), {
|
|
238
|
-
messages: [chatMessage]
|
|
295
|
+
messages: [chatMessage],
|
|
239
296
|
}, {
|
|
240
|
-
headers: {
|
|
241
|
-
'x-api-key': this.apiKey,
|
|
242
|
-
'Content-Type': 'application/json'
|
|
243
|
-
}
|
|
297
|
+
headers: __assign({ "x-api-key": this.apiKey, "Content-Type": "application/json" }, (this.tenant ? { "X-Tenant-Id": this.tenant } : {})),
|
|
244
298
|
})];
|
|
245
299
|
case 2:
|
|
246
300
|
_a.sent();
|
|
247
301
|
return [3 /*break*/, 4];
|
|
248
302
|
case 3:
|
|
249
303
|
error_2 = _a.sent();
|
|
250
|
-
console.error(
|
|
304
|
+
console.error("Error sending message:", error_2);
|
|
251
305
|
// Check if this is the agent inactive error
|
|
252
|
-
if (error_2.response &&
|
|
306
|
+
if (error_2.response &&
|
|
307
|
+
error_2.response.data &&
|
|
308
|
+
error_2.response.data.error_key === "AGENT_INACTIVE") {
|
|
253
309
|
// Create a custom message for the agent inactive error
|
|
254
310
|
if (this.messageHandler) {
|
|
255
311
|
errorMessage = {
|
|
256
312
|
create_time: now,
|
|
257
313
|
start_time: now - this.conversationCreateTime,
|
|
258
|
-
end_time:
|
|
259
|
-
speaker:
|
|
260
|
-
text:
|
|
314
|
+
end_time: now - this.conversationCreateTime + 0.01,
|
|
315
|
+
speaker: "special",
|
|
316
|
+
text: "The agent is currently offline, please check back later. Thank you!",
|
|
261
317
|
};
|
|
262
318
|
this.messageHandler(errorMessage);
|
|
263
319
|
}
|
|
@@ -277,26 +333,23 @@ var ChatService = /** @class */ (function () {
|
|
|
277
333
|
switch (_a.label) {
|
|
278
334
|
case 0:
|
|
279
335
|
if (!this.conversationId) {
|
|
280
|
-
throw new Error(
|
|
336
|
+
throw new Error("Conversation not started");
|
|
281
337
|
}
|
|
282
338
|
formData = new FormData();
|
|
283
|
-
formData.append(
|
|
284
|
-
formData.append(
|
|
339
|
+
formData.append("chat_id", chatId);
|
|
340
|
+
formData.append("file", file);
|
|
285
341
|
_a.label = 1;
|
|
286
342
|
case 1:
|
|
287
343
|
_a.trys.push([1, 3, , 4]);
|
|
288
344
|
return [4 /*yield*/, axios.post("".concat(this.baseUrl, "/api/genagent/knowledge/upload-chat-file"), formData, {
|
|
289
|
-
headers: {
|
|
290
|
-
'x-api-key': this.apiKey,
|
|
291
|
-
'Content-Type': 'multipart/form-data',
|
|
292
|
-
},
|
|
345
|
+
headers: __assign({ "x-api-key": this.apiKey, "Content-Type": "multipart/form-data" }, (this.tenant ? { "X-Tenant-Id": this.tenant } : {})),
|
|
293
346
|
})];
|
|
294
347
|
case 2:
|
|
295
348
|
response = _a.sent();
|
|
296
349
|
return [2 /*return*/, response.data];
|
|
297
350
|
case 3:
|
|
298
351
|
error_3 = _a.sent();
|
|
299
|
-
console.error(
|
|
352
|
+
console.error("Error uploading file:", error_3);
|
|
300
353
|
throw error_3;
|
|
301
354
|
case 4: return [2 /*return*/];
|
|
302
355
|
}
|
|
@@ -309,47 +362,53 @@ var ChatService = /** @class */ (function () {
|
|
|
309
362
|
this.webSocket.close();
|
|
310
363
|
}
|
|
311
364
|
if (!this.conversationId) {
|
|
312
|
-
throw new Error(
|
|
365
|
+
throw new Error("Conversation ID is required for WebSocket connection");
|
|
313
366
|
}
|
|
314
367
|
if (this.connectionStateHandler)
|
|
315
|
-
this.connectionStateHandler(
|
|
316
|
-
var wsUrl = "".concat(this.baseUrl.replace(
|
|
368
|
+
this.connectionStateHandler("connecting");
|
|
369
|
+
var wsUrl = "".concat(this.baseUrl.replace("http", "ws"), "/api/conversations/ws/").concat(this.conversationId, "?api_key=").concat(this.apiKey, "&lang=en&topics=message&topics=takeover&topics=finalize");
|
|
317
370
|
this.webSocket = new WebSocket(wsUrl);
|
|
318
371
|
this.webSocket.onopen = function () {
|
|
319
|
-
console.log(
|
|
372
|
+
console.log("WebSocket connected");
|
|
320
373
|
if (_this.connectionStateHandler)
|
|
321
|
-
_this.connectionStateHandler(
|
|
374
|
+
_this.connectionStateHandler("connected");
|
|
322
375
|
};
|
|
323
376
|
this.webSocket.onmessage = function (event) {
|
|
324
|
-
console.log(
|
|
377
|
+
console.log("WebSocket message:", event.data);
|
|
325
378
|
try {
|
|
326
379
|
var data = JSON.parse(event.data);
|
|
327
|
-
if (data.type ===
|
|
380
|
+
if (data.type === "message" && _this.messageHandler) {
|
|
328
381
|
if (Array.isArray(data.payload)) {
|
|
329
382
|
var messages = data.payload;
|
|
330
383
|
// Adjust timestamps to be relative to conversation start
|
|
331
|
-
var adjustedMessages = messages
|
|
384
|
+
var adjustedMessages = messages
|
|
385
|
+
.map(function (msg) { return _this.adjustMessageTimestamps(msg); })
|
|
386
|
+
.filter(function (msg) { return msg.speaker !== "customer"; });
|
|
332
387
|
adjustedMessages.forEach(_this.messageHandler);
|
|
333
388
|
}
|
|
334
389
|
else {
|
|
335
390
|
var adjustedMessage = _this.adjustMessageTimestamps(data.payload);
|
|
336
|
-
if (adjustedMessage.speaker !==
|
|
391
|
+
if (adjustedMessage.speaker !== "customer") {
|
|
337
392
|
_this.messageHandler(adjustedMessage);
|
|
338
393
|
}
|
|
339
394
|
}
|
|
340
395
|
}
|
|
341
|
-
else if (data.type ===
|
|
396
|
+
else if (data.type === "takeover") {
|
|
342
397
|
// Handle takeover event
|
|
343
|
-
console.log(
|
|
398
|
+
console.log("Takeover event received");
|
|
344
399
|
// Create special message for the takeover indicator
|
|
345
400
|
if (_this.messageHandler) {
|
|
346
401
|
var now = Date.now() / 1000;
|
|
347
402
|
var takeoverMessage = {
|
|
348
403
|
create_time: now,
|
|
349
|
-
start_time: _this.conversationCreateTime
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
404
|
+
start_time: _this.conversationCreateTime
|
|
405
|
+
? now - _this.conversationCreateTime
|
|
406
|
+
: 0,
|
|
407
|
+
end_time: _this.conversationCreateTime
|
|
408
|
+
? now - _this.conversationCreateTime + 0.01
|
|
409
|
+
: 0.01,
|
|
410
|
+
speaker: "special",
|
|
411
|
+
text: "Supervisor took over",
|
|
353
412
|
};
|
|
354
413
|
_this.messageHandler(takeoverMessage);
|
|
355
414
|
}
|
|
@@ -358,18 +417,22 @@ var ChatService = /** @class */ (function () {
|
|
|
358
417
|
_this.takeoverHandler();
|
|
359
418
|
}
|
|
360
419
|
}
|
|
361
|
-
else if (data.type ===
|
|
420
|
+
else if (data.type === "finalize") {
|
|
362
421
|
// Handle finalized event
|
|
363
|
-
console.log(
|
|
422
|
+
console.log("Finalized event received");
|
|
364
423
|
// Create special message for the finalized indicator
|
|
365
424
|
if (_this.messageHandler) {
|
|
366
425
|
var now = Date.now() / 1000;
|
|
367
426
|
var finalizedMessage = {
|
|
368
427
|
create_time: now,
|
|
369
|
-
start_time: _this.conversationCreateTime
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
428
|
+
start_time: _this.conversationCreateTime
|
|
429
|
+
? now - _this.conversationCreateTime
|
|
430
|
+
: 0,
|
|
431
|
+
end_time: _this.conversationCreateTime
|
|
432
|
+
? now - _this.conversationCreateTime + 0.01
|
|
433
|
+
: 0.01,
|
|
434
|
+
speaker: "special",
|
|
435
|
+
text: "Conversation Finalized",
|
|
373
436
|
};
|
|
374
437
|
_this.messageHandler(finalizedMessage);
|
|
375
438
|
}
|
|
@@ -382,18 +445,18 @@ var ChatService = /** @class */ (function () {
|
|
|
382
445
|
}
|
|
383
446
|
}
|
|
384
447
|
catch (error) {
|
|
385
|
-
console.error(
|
|
448
|
+
console.error("Error parsing WebSocket message:", error);
|
|
386
449
|
}
|
|
387
450
|
};
|
|
388
451
|
this.webSocket.onerror = function (error) {
|
|
389
|
-
console.error(
|
|
452
|
+
console.error("WebSocket error:", error);
|
|
390
453
|
if (_this.connectionStateHandler)
|
|
391
|
-
_this.connectionStateHandler(
|
|
454
|
+
_this.connectionStateHandler("disconnected");
|
|
392
455
|
};
|
|
393
456
|
this.webSocket.onclose = function (event) {
|
|
394
|
-
console.log(
|
|
457
|
+
console.log("WebSocket closed:", event.code, event.reason);
|
|
395
458
|
if (_this.connectionStateHandler)
|
|
396
|
-
_this.connectionStateHandler(
|
|
459
|
+
_this.connectionStateHandler("disconnected");
|
|
397
460
|
};
|
|
398
461
|
};
|
|
399
462
|
ChatService.prototype.disconnect = function () {
|
|
@@ -402,6 +465,38 @@ var ChatService = /** @class */ (function () {
|
|
|
402
465
|
this.webSocket = null;
|
|
403
466
|
}
|
|
404
467
|
};
|
|
468
|
+
// Add feedback to a specific agent message in the conversation
|
|
469
|
+
ChatService.prototype.addFeedback = function (messageId, feedback, feedback_message) {
|
|
470
|
+
return __awaiter(this, void 0, void 0, function () {
|
|
471
|
+
var error_4;
|
|
472
|
+
return __generator(this, function (_a) {
|
|
473
|
+
switch (_a.label) {
|
|
474
|
+
case 0:
|
|
475
|
+
if (!this.conversationId) {
|
|
476
|
+
throw new Error("Conversation not started");
|
|
477
|
+
}
|
|
478
|
+
_a.label = 1;
|
|
479
|
+
case 1:
|
|
480
|
+
_a.trys.push([1, 3, , 4]);
|
|
481
|
+
return [4 /*yield*/, axios.patch("".concat(this.baseUrl, "/api/conversations/message/add-feedback/").concat(this.conversationId), {
|
|
482
|
+
message_id: messageId,
|
|
483
|
+
feedback: feedback,
|
|
484
|
+
feedback_message: feedback_message,
|
|
485
|
+
}, {
|
|
486
|
+
headers: __assign({ "x-api-key": this.apiKey, "Content-Type": "application/json" }, (this.tenant ? { "X-Tenant-Id": this.tenant } : {})),
|
|
487
|
+
})];
|
|
488
|
+
case 2:
|
|
489
|
+
_a.sent();
|
|
490
|
+
return [3 /*break*/, 4];
|
|
491
|
+
case 3:
|
|
492
|
+
error_4 = _a.sent();
|
|
493
|
+
console.error("Error adding message feedback:", error_4);
|
|
494
|
+
throw error_4;
|
|
495
|
+
case 4: return [2 /*return*/];
|
|
496
|
+
}
|
|
497
|
+
});
|
|
498
|
+
});
|
|
499
|
+
};
|
|
405
500
|
// Helper method to adjust message timestamps relative to conversation start
|
|
406
501
|
ChatService.prototype.adjustMessageTimestamps = function (message) {
|
|
407
502
|
if (!this.conversationCreateTime) {
|
|
@@ -412,4 +507,4 @@ var ChatService = /** @class */ (function () {
|
|
|
412
507
|
return ChatService;
|
|
413
508
|
}());
|
|
414
509
|
export { ChatService };
|
|
415
|
-
//# 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 { w3cwebsocket as WebSocket, IMessageEvent, ICloseEvent } from 'websocket';\nimport { ChatMessage, StartConversationResponse, Attachment } from '../types';\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 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: ((state: 'connecting' | 'connected' | 'disconnected') => void) | null = null;\n  private storageKey = 'genassist_conversation';\n  private possibleQueries: string[] = [];\n\n  constructor(baseUrl: string, apiKey: string, metadata?: Record<string, any>) {\n    this.baseUrl = baseUrl.endsWith('/') ? baseUrl.slice(0, -1) : baseUrl;\n    this.apiKey = apiKey;\n    this.metadata = metadata;\n    // Try to load a saved conversation ID from localStorage\n    this.loadSavedConversation();\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(handler: (state: 'connecting' | 'connected' | 'disconnected') => void) {\n    this.connectionStateHandler = handler;\n  }\n\n  getPossibleQueries(): string[] {\n    return this.possibleQueries;\n  }\n\n  /**\n   * Load a saved conversation ID from localStorage\n   */\n  private loadSavedConversation(): void {\n    try {\n      const savedConversation = localStorage.getItem(this.storageKey);\n      if (savedConversation) {\n        const { conversationId, createTime, isFinalized } = JSON.parse(savedConversation);\n        this.conversationId = conversationId;\n        this.conversationCreateTime = createTime;\n        this.isFinalized = isFinalized || false;\n        console.log('Loaded saved conversation:', this.conversationId, 'Finalized:', this.isFinalized);\n      }\n    } catch (error) {\n      console.error('Error loading saved conversation:', error);\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        };\n        localStorage.setItem(this.storageKey, JSON.stringify(conversationData));\n        console.log('Saved conversation:', this.conversationId);\n      }\n    } catch (error) {\n      console.error('Error saving conversation:', error);\n    }\n  }\n\n  /**\n   * Reset the current conversation by clearing the ID and websocket\n   */\n  resetConversation(): 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.isFinalized = false;\n\n    // Clear possible queries\n    this.possibleQueries = [];\n    \n    // Remove from local storage\n    try {\n      localStorage.removeItem(this.storageKey);\n    } catch (error) {\n      console.error('Error removing conversation from storage:', error);\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  isConversationFinalized(): boolean {\n    return this.isFinalized;\n  }\n\n  async startConversation(): 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 (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: {\n            'x-api-key': this.apiKey,\n            'Content-Type': 'application/json'\n          }\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 ? response.data.create_time / 1000 : Date.now() / 1000;\n      this.isFinalized = false;\n      this.saveConversation();\n      this.connectWebSocket();\n\n      // Store possible queries if available\n      if (response.data.agent_possible_queries && response.data.agent_possible_queries.length > 0) {\n        this.possibleQueries = response.data.agent_possible_queries;\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.messageHandler(welcomeMessage);\n      }\n      return response.data.conversation_id;\n    } catch (error) {\n      console.error('Error starting conversation:', error);\n      throw error;\n    }\n  }\n\n  async sendMessage(message: string, attachments?: Attachment[]): 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      await axios.patch(\n        `${this.baseUrl}/api/conversations/in-progress/update/${this.conversationId}`,\n        {\n          messages: [chatMessage]\n        },\n        {\n          headers: {\n            'x-api-key': this.apiKey,\n            'Content-Type': 'application/json'\n          }\n        }\n      );\n    } catch (error: any) {\n      console.error('Error sending message:', error);\n      \n      // Check if this is the agent inactive error\n      if (error.response && error.response.data && error.response.data.error_key === 'AGENT_INACTIVE') {\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: 'The agent is currently offline, please check back later. Thank you!'\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<{ fileUrl: string }> {\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<{ fileUrl: string }>(\n        `${this.baseUrl}/api/genagent/knowledge/upload-chat-file`,\n        formData,\n        {\n          headers: {\n            'x-api-key': this.apiKey,\n            'Content-Type': 'multipart/form-data',\n          },\n        }\n      );\n      return response.data;\n    } catch (error) {\n      console.error('Error uploading file:', error);\n      throw error;\n    }\n  }\n\n  connectWebSocket(): void {\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    const wsUrl = `${this.baseUrl.replace('http', 'ws')}/api/conversations/ws/${this.conversationId}?api_key=${this.apiKey}&lang=en&topics=message&topics=takeover&topics=finalize`;\n    this.webSocket = new WebSocket(wsUrl);\n\n    this.webSocket.onopen = () => {\n      console.log('WebSocket connected');\n      if (this.connectionStateHandler) this.connectionStateHandler('connected');\n    };\n\n    this.webSocket.onmessage = (event: IMessageEvent) => {\n      console.log('WebSocket message:', event.data);\n      try {\n        const data = JSON.parse(event.data as string);\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            const adjustedMessages = messages.map(msg => this.adjustMessageTimestamps(msg)).filter(msg => msg.speaker !== 'customer');\n            adjustedMessages.forEach(this.messageHandler);\n          } else {\n            const adjustedMessage = this.adjustMessageTimestamps(data.payload as ChatMessage);\n            if (adjustedMessage.speaker !== 'customer') {\n              this.messageHandler(adjustedMessage);\n            }\n          }\n        } else if (data.type === 'takeover') {\n          // Handle takeover event\n          console.log('Takeover event received');\n          \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 ? now - this.conversationCreateTime : 0,\n              end_time: this.conversationCreateTime ? (now - this.conversationCreateTime) + 0.01 : 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          console.log('Finalized event received');\n          \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 ? now - this.conversationCreateTime : 0,\n              end_time: this.conversationCreateTime ? (now - this.conversationCreateTime) + 0.01 : 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        console.error('Error parsing WebSocket message:', error);\n      }\n    };\n\n    this.webSocket.onerror = (error: Error) => {\n      console.error('WebSocket error:', error);\n      if (this.connectionStateHandler) this.connectionStateHandler('disconnected');\n    };\n\n    this.webSocket.onclose = (event: ICloseEvent) => {\n      console.log('WebSocket closed:', event.code, event.reason);\n      if (this.connectionStateHandler) this.connectionStateHandler('disconnected');\n    };\n  }\n\n  disconnect(): void {\n    if (this.webSocket) {\n      this.webSocket.close();\n      this.webSocket = null;\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} "]}
|
|
510
|
+
//# 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  w3cwebsocket as WebSocket,\n  IMessageEvent,\n  ICloseEvent,\n} from \"websocket\";\nimport {\n  ChatMessage,\n  StartConversationResponse,\n  Attachment,\n  AgentThinkingConfig,\n  AgentWelcomeData,\n} from \"../types\";\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 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 storageKey = \"genassist_conversation\";\n  private possibleQueries: string[] = [];\n  private welcomeData: AgentWelcomeData = {};\n  private thinkingConfig: AgentThinkingConfig = { phrases: [], delayMs: 1000 };\n  private welcomeObjectUrl: string | null = null; // to revoke on reset\n  private tenant: string | undefined;\n\n  constructor(\n    baseUrl: string,\n    apiKey: string,\n    metadata?: Record<string, any>,\n    tenant?: string\n  ) {\n    this.baseUrl = baseUrl.endsWith(\"/\") ? baseUrl.slice(0, -1) : baseUrl;\n    this.apiKey = apiKey;\n    this.metadata = metadata;\n    this.tenant = tenant;\n    // Try to load a saved conversation ID from localStorage\n    this.loadSavedConversation();\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  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   * Load a saved conversation ID from localStorage\n   */\n  private loadSavedConversation(): void {\n    try {\n      const savedConversation = localStorage.getItem(this.storageKey);\n      if (savedConversation) {\n        const { conversationId, createTime, isFinalized } =\n          JSON.parse(savedConversation);\n        this.conversationId = conversationId;\n        this.conversationCreateTime = createTime;\n        this.isFinalized = isFinalized || false;\n        console.log(\n          \"Loaded saved conversation:\",\n          this.conversationId,\n          \"Finalized:\",\n          this.isFinalized\n        );\n      }\n    } catch (error) {\n      console.error(\"Error loading saved conversation:\", error);\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        };\n        localStorage.setItem(this.storageKey, JSON.stringify(conversationData));\n        console.log(\"Saved conversation:\", this.conversationId);\n      }\n    } catch (error) {\n      console.error(\"Error saving conversation:\", error);\n    }\n  }\n\n  /**\n   * Reset the current conversation by clearing the ID and websocket\n   */\n  resetConversation(): 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.isFinalized = false;\n\n    // Clear possible queries\n    this.possibleQueries = [];\n    this.welcomeData = {};\n    this.thinkingConfig = { phrases: [], delayMs: 1000 };\n    if (this.welcomeObjectUrl) {\n      try {\n        URL.revokeObjectURL(this.welcomeObjectUrl);\n      } catch {}\n      this.welcomeObjectUrl = null;\n    }\n\n    // Remove from local storage\n    try {\n      localStorage.removeItem(this.storageKey);\n    } catch (error) {\n      console.error(\"Error removing conversation from storage:\", error);\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  isConversationFinalized(): boolean {\n    return this.isFinalized;\n  }\n\n  async startConversation(): 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 (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: {\n            \"x-api-key\": this.apiKey,\n            \"Content-Type\": \"application/json\",\n            ...(this.tenant ? { \"X-Tenant-Id\": this.tenant } : {}),\n          },\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      this.isFinalized = false;\n      this.saveConversation();\n      this.connectWebSocket();\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;\n      }\n\n      const anyData: any = response.data as any;\n      const agentId: string | undefined = anyData.agent_id;\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        try {\n          const imageResponse = await axios.get(\n            `${this.baseUrl}/api/genagent/agents/configs/${agentId}/welcome-image`,\n            {\n              headers: {\n                \"x-api-key\": this.apiKey,\n                ...(this.tenant ? { \"X-Tenant-Id\": this.tenant } : {}),\n              },\n              responseType: \"blob\",\n            }\n          );\n          const blobUrl = URL.createObjectURL(imageResponse.data);\n          this.welcomeObjectUrl = blobUrl;\n          this.welcomeData.imageUrl = blobUrl;\n        } catch (err) {\n          console.log(\"No welcome image available or failed to load.\");\n        }\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      return response.data.conversation_id;\n    } catch (error) {\n      console.error(\"Error starting conversation:\", error);\n      throw error;\n    }\n  }\n\n  async sendMessage(\n    message: string,\n    attachments?: Attachment[]\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      await axios.patch(\n        `${this.baseUrl}/api/conversations/in-progress/update/${this.conversationId}`,\n        {\n          messages: [chatMessage],\n        },\n        {\n          headers: {\n            \"x-api-key\": this.apiKey,\n            \"Content-Type\": \"application/json\",\n            ...(this.tenant ? { \"X-Tenant-Id\": this.tenant } : {}),\n          },\n        }\n      );\n    } catch (error: any) {\n      console.error(\"Error sending message:\", error);\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: \"The agent is currently offline, please check back later. Thank you!\",\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<{ fileUrl: string }> {\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<{ fileUrl: string }>(\n        `${this.baseUrl}/api/genagent/knowledge/upload-chat-file`,\n        formData,\n        {\n          headers: {\n            \"x-api-key\": this.apiKey,\n            \"Content-Type\": \"multipart/form-data\",\n            ...(this.tenant ? { \"X-Tenant-Id\": this.tenant } : {}),\n          },\n        }\n      );\n      return response.data;\n    } catch (error) {\n      console.error(\"Error uploading file:\", error);\n      throw error;\n    }\n  }\n\n  connectWebSocket(): void {\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    const wsUrl = `${this.baseUrl.replace(\"http\", \"ws\")}/api/conversations/ws/${\n      this.conversationId\n    }?api_key=${\n      this.apiKey\n    }&lang=en&topics=message&topics=takeover&topics=finalize`;\n    this.webSocket = new WebSocket(wsUrl);\n\n    this.webSocket.onopen = () => {\n      console.log(\"WebSocket connected\");\n      if (this.connectionStateHandler) this.connectionStateHandler(\"connected\");\n    };\n\n    this.webSocket.onmessage = (event: IMessageEvent) => {\n      console.log(\"WebSocket message:\", event.data);\n      try {\n        const data = JSON.parse(event.data as string);\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            const adjustedMessages = messages\n              .map((msg) => this.adjustMessageTimestamps(msg))\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            if (adjustedMessage.speaker !== \"customer\") {\n              this.messageHandler(adjustedMessage);\n            }\n          }\n        } else if (data.type === \"takeover\") {\n          // Handle takeover event\n          console.log(\"Takeover event received\");\n\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          console.log(\"Finalized event received\");\n\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        console.error(\"Error parsing WebSocket message:\", error);\n      }\n    };\n\n    this.webSocket.onerror = (error: Error) => {\n      console.error(\"WebSocket error:\", error);\n      if (this.connectionStateHandler)\n        this.connectionStateHandler(\"disconnected\");\n    };\n\n    this.webSocket.onclose = (event: ICloseEvent) => {\n      console.log(\"WebSocket closed:\", event.code, event.reason);\n      if (this.connectionStateHandler)\n        this.connectionStateHandler(\"disconnected\");\n    };\n  }\n\n  disconnect(): void {\n    if (this.webSocket) {\n      this.webSocket.close();\n      this.webSocket = null;\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 (!this.conversationId) {\n      throw new Error(\"Conversation not started\");\n    }\n\n    try {\n      await axios.patch(\n        `${this.baseUrl}/api/conversations/message/add-feedback/${this.conversationId}`,\n        {\n          message_id: messageId,\n          feedback,\n          feedback_message,\n        },\n        {\n          headers: {\n            \"x-api-key\": this.apiKey,\n            \"Content-Type\": \"application/json\",\n            ...(this.tenant ? { \"X-Tenant-Id\": this.tenant } : {}),\n          },\n        }\n      );\n    } catch (error) {\n      console.error(\"Error adding message feedback:\", error);\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"]}
|