genassist-chat-react 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,118 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ var __generator = (this && this.__generator) || function (thisArg, body) {
11
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
12
+ return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
13
+ function verb(n) { return function (v) { return step([n, v]); }; }
14
+ function step(op) {
15
+ if (f) throw new TypeError("Generator is already executing.");
16
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
17
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
18
+ if (y = 0, t) op = [op[0] & 2, t.value];
19
+ switch (op[0]) {
20
+ case 0: case 1: t = op; break;
21
+ case 4: _.label++; return { value: op[1], done: false };
22
+ case 5: _.label++; y = op[1]; op = [0]; continue;
23
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
24
+ default:
25
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
26
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
27
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
28
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
29
+ if (t[2]) _.ops.pop();
30
+ _.trys.pop(); continue;
31
+ }
32
+ op = body.call(thisArg, _);
33
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
34
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
35
+ }
36
+ };
37
+ var AudioService = /** @class */ (function () {
38
+ function AudioService(config) {
39
+ this.ws = null;
40
+ this.audioChunks = [];
41
+ this.resolvePromise = null;
42
+ this.rejectPromise = null;
43
+ this.baseUrl = config.baseUrl;
44
+ this.apiKey = config.apiKey;
45
+ }
46
+ AudioService.prototype.textToSpeech = function (text_1) {
47
+ return __awaiter(this, arguments, void 0, function (text, voice) {
48
+ var _this = this;
49
+ if (voice === void 0) { voice = 'alloy'; }
50
+ return __generator(this, function (_a) {
51
+ return [2 /*return*/, new Promise(function (resolve, reject) {
52
+ _this.resolvePromise = resolve;
53
+ _this.rejectPromise = reject;
54
+ _this.audioChunks = [];
55
+ // Create WebSocket connection
56
+ var wsUrl = "".concat(_this.baseUrl.replace('http', 'ws'), "/api/voice/audio/tts?api_key=").concat(_this.apiKey);
57
+ _this.ws = new WebSocket(wsUrl);
58
+ _this.ws.onopen = function () {
59
+ var _a;
60
+ (_a = _this.ws) === null || _a === void 0 ? void 0 : _a.send(JSON.stringify({ text: text }));
61
+ };
62
+ _this.ws.onmessage = function (event) {
63
+ if (event.data instanceof Blob) {
64
+ _this.audioChunks.push(event.data);
65
+ }
66
+ };
67
+ _this.ws.onclose = function () {
68
+ var _a, _b;
69
+ if (_this.audioChunks.length > 0) {
70
+ var audioBlob = new Blob(_this.audioChunks, { type: 'audio/mp3' });
71
+ (_a = _this.resolvePromise) === null || _a === void 0 ? void 0 : _a.call(_this, audioBlob);
72
+ }
73
+ else {
74
+ (_b = _this.rejectPromise) === null || _b === void 0 ? void 0 : _b.call(_this, new Error('No audio data received'));
75
+ }
76
+ _this.cleanup();
77
+ };
78
+ _this.ws.onerror = function (error) {
79
+ var _a;
80
+ (_a = _this.rejectPromise) === null || _a === void 0 ? void 0 : _a.call(_this, error);
81
+ _this.cleanup();
82
+ };
83
+ })];
84
+ });
85
+ });
86
+ };
87
+ AudioService.prototype.cleanup = function () {
88
+ if (this.ws) {
89
+ this.ws.close();
90
+ this.ws = null;
91
+ }
92
+ this.resolvePromise = null;
93
+ this.rejectPromise = null;
94
+ };
95
+ AudioService.prototype.playAudio = function (audioBlob) {
96
+ return __awaiter(this, void 0, void 0, function () {
97
+ var audioUrl, audio;
98
+ return __generator(this, function (_a) {
99
+ audioUrl = URL.createObjectURL(audioBlob);
100
+ audio = new Audio(audioUrl);
101
+ return [2 /*return*/, new Promise(function (resolve, reject) {
102
+ audio.onended = function () {
103
+ URL.revokeObjectURL(audioUrl);
104
+ resolve();
105
+ };
106
+ audio.onerror = function (error) {
107
+ URL.revokeObjectURL(audioUrl);
108
+ reject(error);
109
+ };
110
+ audio.play().catch(reject);
111
+ })];
112
+ });
113
+ });
114
+ };
115
+ return AudioService;
116
+ }());
117
+ export { AudioService };
118
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXVkaW9TZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NlcnZpY2VzL2F1ZGlvU2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFPQTtJQVFFLHNCQUFZLE1BQTBCO1FBTDlCLE9BQUUsR0FBcUIsSUFBSSxDQUFDO1FBQzVCLGdCQUFXLEdBQVcsRUFBRSxDQUFDO1FBQ3pCLG1CQUFjLEdBQW1DLElBQUksQ0FBQztRQUN0RCxrQkFBYSxHQUFvQyxJQUFJLENBQUM7UUFHNUQsSUFBSSxDQUFDLE9BQU8sR0FBRyxNQUFNLENBQUMsT0FBTyxDQUFDO1FBQzlCLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFDLE1BQU0sQ0FBQztJQUM5QixDQUFDO0lBRUssbUNBQVksR0FBbEI7NERBQW1CLElBQVksRUFBRSxLQUF1Qjs7WUFBdkIsc0JBQUEsRUFBQSxlQUF1Qjs7Z0JBQ3RELHNCQUFPLElBQUksT0FBTyxDQUFDLFVBQUMsT0FBTyxFQUFFLE1BQU07d0JBQ2pDLEtBQUksQ0FBQyxjQUFjLEdBQUcsT0FBTyxDQUFDO3dCQUM5QixLQUFJLENBQUMsYUFBYSxHQUFHLE1BQU0sQ0FBQzt3QkFDNUIsS0FBSSxDQUFDLFdBQVcsR0FBRyxFQUFFLENBQUM7d0JBRXRCLDhCQUE4Qjt3QkFDOUIsSUFBTSxLQUFLLEdBQUcsVUFBRyxLQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxNQUFNLEVBQUUsSUFBSSxDQUFDLDBDQUFnQyxLQUFJLENBQUMsTUFBTSxDQUFFLENBQUM7d0JBQ2pHLEtBQUksQ0FBQyxFQUFFLEdBQUcsSUFBSSxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUM7d0JBRS9CLEtBQUksQ0FBQyxFQUFFLENBQUMsTUFBTSxHQUFHOzs0QkFDZixNQUFBLEtBQUksQ0FBQyxFQUFFLDBDQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLEVBQUUsSUFBSSxNQUFBLEVBQUUsQ0FBQyxDQUFDLENBQUM7d0JBQzFDLENBQUMsQ0FBQzt3QkFFRixLQUFJLENBQUMsRUFBRSxDQUFDLFNBQVMsR0FBRyxVQUFDLEtBQUs7NEJBQ3hCLElBQUksS0FBSyxDQUFDLElBQUksWUFBWSxJQUFJLEVBQUUsQ0FBQztnQ0FDL0IsS0FBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDOzRCQUNwQyxDQUFDO3dCQUNILENBQUMsQ0FBQzt3QkFFRixLQUFJLENBQUMsRUFBRSxDQUFDLE9BQU8sR0FBRzs7NEJBQ2hCLElBQUksS0FBSSxDQUFDLFdBQVcsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7Z0NBQ2hDLElBQU0sU0FBUyxHQUFHLElBQUksSUFBSSxDQUFDLEtBQUksQ0FBQyxXQUFXLEVBQUUsRUFBRSxJQUFJLEVBQUUsV0FBVyxFQUFFLENBQUMsQ0FBQztnQ0FDcEUsTUFBQSxLQUFJLENBQUMsY0FBYyxzREFBRyxTQUFTLENBQUMsQ0FBQzs0QkFDbkMsQ0FBQztpQ0FBTSxDQUFDO2dDQUNOLE1BQUEsS0FBSSxDQUFDLGFBQWEsc0RBQUcsSUFBSSxLQUFLLENBQUMsd0JBQXdCLENBQUMsQ0FBQyxDQUFDOzRCQUM1RCxDQUFDOzRCQUNELEtBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQzt3QkFDakIsQ0FBQyxDQUFDO3dCQUVGLEtBQUksQ0FBQyxFQUFFLENBQUMsT0FBTyxHQUFHLFVBQUMsS0FBSzs7NEJBQ3RCLE1BQUEsS0FBSSxDQUFDLGFBQWEsc0RBQUcsS0FBSyxDQUFDLENBQUM7NEJBQzVCLEtBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQzt3QkFDakIsQ0FBQyxDQUFDO29CQUNKLENBQUMsQ0FBQyxFQUFDOzs7S0FDSjtJQUVPLDhCQUFPLEdBQWY7UUFDRSxJQUFJLElBQUksQ0FBQyxFQUFFLEVBQUUsQ0FBQztZQUNaLElBQUksQ0FBQyxFQUFFLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDaEIsSUFBSSxDQUFDLEVBQUUsR0FBRyxJQUFJLENBQUM7UUFDakIsQ0FBQztRQUNELElBQUksQ0FBQyxjQUFjLEdBQUcsSUFBSSxDQUFDO1FBQzNCLElBQUksQ0FBQyxhQUFhLEdBQUcsSUFBSSxDQUFDO0lBQzVCLENBQUM7SUFFSyxnQ0FBUyxHQUFmLFVBQWdCLFNBQWU7Ozs7Z0JBQ3ZCLFFBQVEsR0FBRyxHQUFHLENBQUMsZUFBZSxDQUFDLFNBQVMsQ0FBQyxDQUFDO2dCQUMxQyxLQUFLLEdBQUcsSUFBSSxLQUFLLENBQUMsUUFBUSxDQUFDLENBQUM7Z0JBRWxDLHNCQUFPLElBQUksT0FBTyxDQUFDLFVBQUMsT0FBTyxFQUFFLE1BQU07d0JBQ2pDLEtBQUssQ0FBQyxPQUFPLEdBQUc7NEJBQ2QsR0FBRyxDQUFDLGVBQWUsQ0FBQyxRQUFRLENBQUMsQ0FBQzs0QkFDOUIsT0FBTyxFQUFFLENBQUM7d0JBQ1osQ0FBQyxDQUFDO3dCQUVGLEtBQUssQ0FBQyxPQUFPLEdBQUcsVUFBQyxLQUFLOzRCQUNwQixHQUFHLENBQUMsZUFBZSxDQUFDLFFBQVEsQ0FBQyxDQUFDOzRCQUM5QixNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7d0JBQ2hCLENBQUMsQ0FBQzt3QkFFRixLQUFLLENBQUMsSUFBSSxFQUFFLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxDQUFDO29CQUM3QixDQUFDLENBQUMsRUFBQzs7O0tBQ0o7SUFDSCxtQkFBQztBQUFELENBQUMsQUE3RUQsSUE2RUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgYXhpb3MgZnJvbSAnYXhpb3MnO1xuXG5pbnRlcmZhY2UgQXVkaW9TZXJ2aWNlQ29uZmlnIHtcbiAgYmFzZVVybDogc3RyaW5nO1xuICBhcGlLZXk6IHN0cmluZztcbn1cblxuZXhwb3J0IGNsYXNzIEF1ZGlvU2VydmljZSB7XG4gIHByaXZhdGUgYmFzZVVybDogc3RyaW5nO1xuICBwcml2YXRlIGFwaUtleTogc3RyaW5nO1xuICBwcml2YXRlIHdzOiBXZWJTb2NrZXQgfCBudWxsID0gbnVsbDtcbiAgcHJpdmF0ZSBhdWRpb0NodW5rczogQmxvYltdID0gW107XG4gIHByaXZhdGUgcmVzb2x2ZVByb21pc2U6ICgodmFsdWU6IEJsb2IpID0+IHZvaWQpIHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgcmVqZWN0UHJvbWlzZTogKChyZWFzb24/OiBhbnkpID0+IHZvaWQpIHwgbnVsbCA9IG51bGw7XG5cbiAgY29uc3RydWN0b3IoY29uZmlnOiBBdWRpb1NlcnZpY2VDb25maWcpIHtcbiAgICB0aGlzLmJhc2VVcmwgPSBjb25maWcuYmFzZVVybDtcbiAgICB0aGlzLmFwaUtleSA9IGNvbmZpZy5hcGlLZXk7XG4gIH1cblxuICBhc3luYyB0ZXh0VG9TcGVlY2godGV4dDogc3RyaW5nLCB2b2ljZTogc3RyaW5nID0gJ2FsbG95Jyk6IFByb21pc2U8QmxvYj4ge1xuICAgIHJldHVybiBuZXcgUHJvbWlzZSgocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICB0aGlzLnJlc29sdmVQcm9taXNlID0gcmVzb2x2ZTtcbiAgICAgIHRoaXMucmVqZWN0UHJvbWlzZSA9IHJlamVjdDtcbiAgICAgIHRoaXMuYXVkaW9DaHVua3MgPSBbXTtcblxuICAgICAgLy8gQ3JlYXRlIFdlYlNvY2tldCBjb25uZWN0aW9uXG4gICAgICBjb25zdCB3c1VybCA9IGAke3RoaXMuYmFzZVVybC5yZXBsYWNlKCdodHRwJywgJ3dzJyl9L2FwaS92b2ljZS9hdWRpby90dHM/YXBpX2tleT0ke3RoaXMuYXBpS2V5fWA7XG4gICAgICB0aGlzLndzID0gbmV3IFdlYlNvY2tldCh3c1VybCk7XG5cbiAgICAgIHRoaXMud3Mub25vcGVuID0gKCkgPT4ge1xuICAgICAgICB0aGlzLndzPy5zZW5kKEpTT04uc3RyaW5naWZ5KHsgdGV4dCB9KSk7XG4gICAgICB9O1xuXG4gICAgICB0aGlzLndzLm9ubWVzc2FnZSA9IChldmVudCkgPT4ge1xuICAgICAgICBpZiAoZXZlbnQuZGF0YSBpbnN0YW5jZW9mIEJsb2IpIHtcbiAgICAgICAgICB0aGlzLmF1ZGlvQ2h1bmtzLnB1c2goZXZlbnQuZGF0YSk7XG4gICAgICAgIH1cbiAgICAgIH07XG5cbiAgICAgIHRoaXMud3Mub25jbG9zZSA9ICgpID0+IHtcbiAgICAgICAgaWYgKHRoaXMuYXVkaW9DaHVua3MubGVuZ3RoID4gMCkge1xuICAgICAgICAgIGNvbnN0IGF1ZGlvQmxvYiA9IG5ldyBCbG9iKHRoaXMuYXVkaW9DaHVua3MsIHsgdHlwZTogJ2F1ZGlvL21wMycgfSk7XG4gICAgICAgICAgdGhpcy5yZXNvbHZlUHJvbWlzZT8uKGF1ZGlvQmxvYik7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgdGhpcy5yZWplY3RQcm9taXNlPy4obmV3IEVycm9yKCdObyBhdWRpbyBkYXRhIHJlY2VpdmVkJykpO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMuY2xlYW51cCgpO1xuICAgICAgfTtcblxuICAgICAgdGhpcy53cy5vbmVycm9yID0gKGVycm9yKSA9PiB7XG4gICAgICAgIHRoaXMucmVqZWN0UHJvbWlzZT8uKGVycm9yKTtcbiAgICAgICAgdGhpcy5jbGVhbnVwKCk7XG4gICAgICB9O1xuICAgIH0pO1xuICB9XG5cbiAgcHJpdmF0ZSBjbGVhbnVwKCkge1xuICAgIGlmICh0aGlzLndzKSB7XG4gICAgICB0aGlzLndzLmNsb3NlKCk7XG4gICAgICB0aGlzLndzID0gbnVsbDtcbiAgICB9XG4gICAgdGhpcy5yZXNvbHZlUHJvbWlzZSA9IG51bGw7XG4gICAgdGhpcy5yZWplY3RQcm9taXNlID0gbnVsbDtcbiAgfVxuXG4gIGFzeW5jIHBsYXlBdWRpbyhhdWRpb0Jsb2I6IEJsb2IpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCBhdWRpb1VybCA9IFVSTC5jcmVhdGVPYmplY3RVUkwoYXVkaW9CbG9iKTtcbiAgICBjb25zdCBhdWRpbyA9IG5ldyBBdWRpbyhhdWRpb1VybCk7XG4gICAgXG4gICAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgIGF1ZGlvLm9uZW5kZWQgPSAoKSA9PiB7XG4gICAgICAgIFVSTC5yZXZva2VPYmplY3RVUkwoYXVkaW9VcmwpO1xuICAgICAgICByZXNvbHZlKCk7XG4gICAgICB9O1xuICAgICAgXG4gICAgICBhdWRpby5vbmVycm9yID0gKGVycm9yKSA9PiB7XG4gICAgICAgIFVSTC5yZXZva2VPYmplY3RVUkwoYXVkaW9VcmwpO1xuICAgICAgICByZWplY3QoZXJyb3IpO1xuICAgICAgfTtcbiAgICAgIFxuICAgICAgYXVkaW8ucGxheSgpLmNhdGNoKHJlamVjdCk7XG4gICAgfSk7XG4gIH1cbn0gIl19
@@ -0,0 +1,48 @@
1
+ import { ChatMessage } from '../types';
2
+ export declare class ChatService {
3
+ private baseUrl;
4
+ private apiKey;
5
+ private metadata;
6
+ private conversationId;
7
+ private conversationCreateTime;
8
+ private isFinalized;
9
+ private webSocket;
10
+ private messageHandler;
11
+ private takeoverHandler;
12
+ private finalizedHandler;
13
+ private connectionStateHandler;
14
+ private storageKey;
15
+ private possibleQueries;
16
+ constructor(baseUrl: string, apiKey: string, metadata?: Record<string, any>);
17
+ setMessageHandler(handler: (message: ChatMessage) => void): void;
18
+ setTakeoverHandler(handler: () => void): void;
19
+ setFinalizedHandler(handler: () => void): void;
20
+ setConnectionStateHandler(handler: (state: 'connecting' | 'connected' | 'disconnected') => void): void;
21
+ getPossibleQueries(): string[];
22
+ /**
23
+ * Load a saved conversation ID from localStorage
24
+ */
25
+ private loadSavedConversation;
26
+ /**
27
+ * Save the current conversation ID to localStorage
28
+ */
29
+ private saveConversation;
30
+ /**
31
+ * Reset the current conversation by clearing the ID and websocket
32
+ */
33
+ resetConversation(): void;
34
+ /**
35
+ * Check if there's a current conversation
36
+ */
37
+ hasActiveConversation(): boolean;
38
+ /**
39
+ * Get the current conversation ID
40
+ */
41
+ getConversationId(): string | null;
42
+ isConversationFinalized(): boolean;
43
+ startConversation(): Promise<string>;
44
+ sendMessage(message: string): Promise<void>;
45
+ connectWebSocket(): void;
46
+ disconnect(): void;
47
+ private adjustMessageTimestamps;
48
+ }
@@ -0,0 +1,376 @@
1
+ var __assign = (this && this.__assign) || function () {
2
+ __assign = Object.assign || function(t) {
3
+ for (var s, i = 1, n = arguments.length; i < n; i++) {
4
+ s = arguments[i];
5
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
6
+ t[p] = s[p];
7
+ }
8
+ return t;
9
+ };
10
+ return __assign.apply(this, arguments);
11
+ };
12
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
13
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
14
+ return new (P || (P = Promise))(function (resolve, reject) {
15
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
16
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
17
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
18
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
19
+ });
20
+ };
21
+ var __generator = (this && this.__generator) || function (thisArg, body) {
22
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
23
+ return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
24
+ function verb(n) { return function (v) { return step([n, v]); }; }
25
+ function step(op) {
26
+ if (f) throw new TypeError("Generator is already executing.");
27
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
28
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
29
+ if (y = 0, t) op = [op[0] & 2, t.value];
30
+ switch (op[0]) {
31
+ case 0: case 1: t = op; break;
32
+ case 4: _.label++; return { value: op[1], done: false };
33
+ case 5: _.label++; y = op[1]; op = [0]; continue;
34
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
35
+ default:
36
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
37
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
38
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
39
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
40
+ if (t[2]) _.ops.pop();
41
+ _.trys.pop(); continue;
42
+ }
43
+ op = body.call(thisArg, _);
44
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
45
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
46
+ }
47
+ };
48
+ import axios from 'axios';
49
+ import { w3cwebsocket as WebSocket } from 'websocket';
50
+ var ChatService = /** @class */ (function () {
51
+ function ChatService(baseUrl, apiKey, metadata) {
52
+ this.conversationId = null;
53
+ this.conversationCreateTime = null; // Track conversation start time
54
+ this.isFinalized = false;
55
+ this.webSocket = null;
56
+ this.messageHandler = null;
57
+ this.takeoverHandler = null;
58
+ this.finalizedHandler = null;
59
+ this.connectionStateHandler = null;
60
+ this.storageKey = 'genassist_conversation';
61
+ this.possibleQueries = [];
62
+ this.baseUrl = baseUrl.endsWith('/') ? baseUrl.slice(0, -1) : baseUrl;
63
+ this.apiKey = apiKey;
64
+ this.metadata = metadata;
65
+ // Try to load a saved conversation ID from localStorage
66
+ this.loadSavedConversation();
67
+ }
68
+ ChatService.prototype.setMessageHandler = function (handler) {
69
+ this.messageHandler = handler;
70
+ };
71
+ ChatService.prototype.setTakeoverHandler = function (handler) {
72
+ this.takeoverHandler = handler;
73
+ };
74
+ ChatService.prototype.setFinalizedHandler = function (handler) {
75
+ this.finalizedHandler = handler;
76
+ };
77
+ ChatService.prototype.setConnectionStateHandler = function (handler) {
78
+ this.connectionStateHandler = handler;
79
+ };
80
+ ChatService.prototype.getPossibleQueries = function () {
81
+ return this.possibleQueries;
82
+ };
83
+ /**
84
+ * Load a saved conversation ID from localStorage
85
+ */
86
+ ChatService.prototype.loadSavedConversation = function () {
87
+ try {
88
+ var savedConversation = localStorage.getItem(this.storageKey);
89
+ if (savedConversation) {
90
+ var _a = JSON.parse(savedConversation), conversationId = _a.conversationId, createTime = _a.createTime, isFinalized = _a.isFinalized;
91
+ this.conversationId = conversationId;
92
+ this.conversationCreateTime = createTime;
93
+ this.isFinalized = isFinalized || false;
94
+ console.log('Loaded saved conversation:', this.conversationId, 'Finalized:', this.isFinalized);
95
+ }
96
+ }
97
+ catch (error) {
98
+ console.error('Error loading saved conversation:', error);
99
+ }
100
+ };
101
+ /**
102
+ * Save the current conversation ID to localStorage
103
+ */
104
+ ChatService.prototype.saveConversation = function () {
105
+ try {
106
+ if (this.conversationId && this.conversationCreateTime) {
107
+ var conversationData = {
108
+ conversationId: this.conversationId,
109
+ createTime: this.conversationCreateTime,
110
+ isFinalized: this.isFinalized,
111
+ };
112
+ localStorage.setItem(this.storageKey, JSON.stringify(conversationData));
113
+ console.log('Saved conversation:', this.conversationId);
114
+ }
115
+ }
116
+ catch (error) {
117
+ console.error('Error saving conversation:', error);
118
+ }
119
+ };
120
+ /**
121
+ * Reset the current conversation by clearing the ID and websocket
122
+ */
123
+ ChatService.prototype.resetConversation = function () {
124
+ // Close the current websocket connection if it exists
125
+ if (this.webSocket) {
126
+ this.webSocket.close();
127
+ this.webSocket = null;
128
+ }
129
+ // Clear the conversation ID
130
+ this.conversationId = null;
131
+ this.conversationCreateTime = null;
132
+ this.isFinalized = false;
133
+ // Clear possible queries
134
+ this.possibleQueries = [];
135
+ // Remove from local storage
136
+ try {
137
+ localStorage.removeItem(this.storageKey);
138
+ }
139
+ catch (error) {
140
+ console.error('Error removing conversation from storage:', error);
141
+ }
142
+ };
143
+ /**
144
+ * Check if there's a current conversation
145
+ */
146
+ ChatService.prototype.hasActiveConversation = function () {
147
+ return !!this.conversationId;
148
+ };
149
+ /**
150
+ * Get the current conversation ID
151
+ */
152
+ ChatService.prototype.getConversationId = function () {
153
+ return this.conversationId;
154
+ };
155
+ ChatService.prototype.isConversationFinalized = function () {
156
+ return this.isFinalized;
157
+ };
158
+ ChatService.prototype.startConversation = function () {
159
+ return __awaiter(this, void 0, void 0, function () {
160
+ var requestBody, response, now, welcomeMessage, error_1;
161
+ return __generator(this, function (_a) {
162
+ switch (_a.label) {
163
+ case 0:
164
+ _a.trys.push([0, 2, , 3]);
165
+ requestBody = {
166
+ messages: [],
167
+ recorded_at: new Date().toISOString(),
168
+ data_source_id: "00000000-0000-0000-0000-000000000000"
169
+ };
170
+ if (this.metadata) {
171
+ requestBody.metadata = this.metadata;
172
+ }
173
+ 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
+ }
178
+ })];
179
+ case 1:
180
+ response = _a.sent();
181
+ this.conversationId = response.data.conversation_id;
182
+ // Store conversation create time (use from response if available, otherwise current time)
183
+ this.conversationCreateTime = response.data.create_time ? response.data.create_time / 1000 : Date.now() / 1000;
184
+ this.isFinalized = false;
185
+ this.saveConversation();
186
+ this.connectWebSocket();
187
+ // Store possible queries if available
188
+ if (response.data.agent_possible_queries && response.data.agent_possible_queries.length > 0) {
189
+ this.possibleQueries = response.data.agent_possible_queries;
190
+ }
191
+ // Process agent welcome message if available
192
+ if (response.data.agent_welcome_message && this.messageHandler) {
193
+ now = Date.now() / 1000;
194
+ welcomeMessage = {
195
+ create_time: now,
196
+ start_time: now - this.conversationCreateTime, // Relative to conversation start
197
+ end_time: (now - this.conversationCreateTime) + 0.01, // Relative to conversation start
198
+ speaker: 'agent',
199
+ text: response.data.agent_welcome_message
200
+ };
201
+ this.messageHandler(welcomeMessage);
202
+ }
203
+ return [2 /*return*/, response.data.conversation_id];
204
+ case 2:
205
+ error_1 = _a.sent();
206
+ console.error('Error starting conversation:', error_1);
207
+ throw error_1;
208
+ case 3: return [2 /*return*/];
209
+ }
210
+ });
211
+ });
212
+ };
213
+ ChatService.prototype.sendMessage = function (message) {
214
+ return __awaiter(this, void 0, void 0, function () {
215
+ var now, chatMessage, error_2, errorMessage;
216
+ return __generator(this, function (_a) {
217
+ switch (_a.label) {
218
+ case 0:
219
+ if (!this.conversationId || !this.conversationCreateTime) {
220
+ throw new Error('Conversation not started');
221
+ }
222
+ now = Date.now() / 1000;
223
+ chatMessage = {
224
+ create_time: now,
225
+ start_time: now - this.conversationCreateTime, // Relative to conversation start
226
+ end_time: (now - this.conversationCreateTime) + 0.01, // Relative to conversation start
227
+ speaker: 'customer',
228
+ text: message
229
+ };
230
+ _a.label = 1;
231
+ case 1:
232
+ _a.trys.push([1, 3, , 4]);
233
+ return [4 /*yield*/, axios.patch("".concat(this.baseUrl, "/api/conversations/in-progress/update/").concat(this.conversationId), {
234
+ messages: [chatMessage]
235
+ }, {
236
+ headers: {
237
+ 'x-api-key': this.apiKey,
238
+ 'Content-Type': 'application/json'
239
+ }
240
+ })];
241
+ case 2:
242
+ _a.sent();
243
+ return [3 /*break*/, 4];
244
+ case 3:
245
+ error_2 = _a.sent();
246
+ console.error('Error sending message:', error_2);
247
+ // Check if this is the agent inactive error
248
+ if (error_2.response && error_2.response.data && error_2.response.data.error_key === 'AGENT_INACTIVE') {
249
+ // Create a custom message for the agent inactive error
250
+ if (this.messageHandler) {
251
+ errorMessage = {
252
+ create_time: now,
253
+ start_time: now - this.conversationCreateTime,
254
+ end_time: (now - this.conversationCreateTime) + 0.01,
255
+ speaker: 'special',
256
+ text: 'The agent is currently offline, please check back later. Thank you!'
257
+ };
258
+ this.messageHandler(errorMessage);
259
+ }
260
+ // Don't throw the error since we handled it with a message
261
+ return [2 /*return*/];
262
+ }
263
+ throw error_2;
264
+ case 4: return [2 /*return*/];
265
+ }
266
+ });
267
+ });
268
+ };
269
+ ChatService.prototype.connectWebSocket = function () {
270
+ var _this = this;
271
+ if (this.webSocket) {
272
+ this.webSocket.close();
273
+ }
274
+ if (!this.conversationId) {
275
+ throw new Error('Conversation ID is required for WebSocket connection');
276
+ }
277
+ if (this.connectionStateHandler)
278
+ this.connectionStateHandler('connecting');
279
+ 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");
280
+ this.webSocket = new WebSocket(wsUrl);
281
+ this.webSocket.onopen = function () {
282
+ console.log('WebSocket connected');
283
+ if (_this.connectionStateHandler)
284
+ _this.connectionStateHandler('connected');
285
+ };
286
+ this.webSocket.onmessage = function (event) {
287
+ console.log('WebSocket message:', event.data);
288
+ try {
289
+ var data = JSON.parse(event.data);
290
+ if (data.type === 'message' && _this.messageHandler) {
291
+ if (Array.isArray(data.payload)) {
292
+ var messages = data.payload;
293
+ // Adjust timestamps to be relative to conversation start
294
+ var adjustedMessages = messages.map(function (msg) { return _this.adjustMessageTimestamps(msg); });
295
+ adjustedMessages.forEach(_this.messageHandler);
296
+ }
297
+ else {
298
+ var adjustedMessage = _this.adjustMessageTimestamps(data.payload);
299
+ _this.messageHandler(adjustedMessage);
300
+ }
301
+ }
302
+ else if (data.type === 'takeover') {
303
+ // Handle takeover event
304
+ console.log('Takeover event received');
305
+ // Create special message for the takeover indicator
306
+ if (_this.messageHandler) {
307
+ var now = Date.now() / 1000;
308
+ var takeoverMessage = {
309
+ create_time: now,
310
+ start_time: _this.conversationCreateTime ? now - _this.conversationCreateTime : 0,
311
+ end_time: _this.conversationCreateTime ? (now - _this.conversationCreateTime) + 0.01 : 0.01,
312
+ speaker: 'special',
313
+ text: 'Supervisor took over'
314
+ };
315
+ _this.messageHandler(takeoverMessage);
316
+ }
317
+ // Call the takeover handler if provided
318
+ if (_this.takeoverHandler) {
319
+ _this.takeoverHandler();
320
+ }
321
+ }
322
+ else if (data.type === 'finalize') {
323
+ // Handle finalized event
324
+ console.log('Finalized event received');
325
+ // Create special message for the finalized indicator
326
+ if (_this.messageHandler) {
327
+ var now = Date.now() / 1000;
328
+ var finalizedMessage = {
329
+ create_time: now,
330
+ start_time: _this.conversationCreateTime ? now - _this.conversationCreateTime : 0,
331
+ end_time: _this.conversationCreateTime ? (now - _this.conversationCreateTime) + 0.01 : 0.01,
332
+ speaker: 'special',
333
+ text: 'Conversation Finalized'
334
+ };
335
+ _this.messageHandler(finalizedMessage);
336
+ }
337
+ // Call the finalized handler if provided
338
+ if (_this.finalizedHandler) {
339
+ _this.finalizedHandler();
340
+ }
341
+ _this.isFinalized = true;
342
+ _this.saveConversation();
343
+ }
344
+ }
345
+ catch (error) {
346
+ console.error('Error parsing WebSocket message:', error);
347
+ }
348
+ };
349
+ this.webSocket.onerror = function (error) {
350
+ console.error('WebSocket error:', error);
351
+ if (_this.connectionStateHandler)
352
+ _this.connectionStateHandler('disconnected');
353
+ };
354
+ this.webSocket.onclose = function (event) {
355
+ console.log('WebSocket closed:', event.code, event.reason);
356
+ if (_this.connectionStateHandler)
357
+ _this.connectionStateHandler('disconnected');
358
+ };
359
+ };
360
+ ChatService.prototype.disconnect = function () {
361
+ if (this.webSocket) {
362
+ this.webSocket.close();
363
+ this.webSocket = null;
364
+ }
365
+ };
366
+ // Helper method to adjust message timestamps relative to conversation start
367
+ ChatService.prototype.adjustMessageTimestamps = function (message) {
368
+ if (!this.conversationCreateTime) {
369
+ return message;
370
+ }
371
+ return __assign(__assign({}, message), { start_time: message.start_time - this.conversationCreateTime, end_time: message.end_time - this.conversationCreateTime });
372
+ };
373
+ return ChatService;
374
+ }());
375
+ export { ChatService };
376
+ //# 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 } 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): 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    };\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  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));\n            adjustedMessages.forEach(this.messageHandler);\n          } else {\n            const adjustedMessage = this.adjustMessageTimestamps(data.payload as ChatMessage);\n            this.messageHandler(adjustedMessage);\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} "]}
@@ -0,0 +1,42 @@
1
+ export interface ChatMessage {
2
+ create_time: number;
3
+ start_time: number;
4
+ end_time: number;
5
+ speaker: 'customer' | 'agent' | 'special';
6
+ text: string;
7
+ }
8
+ export interface StartConversationResponse {
9
+ message: string;
10
+ conversation_id: string;
11
+ agent_welcome_message?: string;
12
+ agent_possible_queries?: string[];
13
+ create_time?: number;
14
+ }
15
+ export interface GenAgentChatProps {
16
+ baseUrl: string;
17
+ apiKey: string;
18
+ metadata?: Record<string, any>;
19
+ onError?: (error: Error) => void;
20
+ onTakeover?: () => void;
21
+ onFinalize?: () => void;
22
+ theme?: {
23
+ primaryColor?: string;
24
+ secondaryColor?: string;
25
+ fontFamily?: string;
26
+ fontSize?: string;
27
+ backgroundColor?: string;
28
+ textColor?: string;
29
+ };
30
+ headerTitle?: string;
31
+ placeholder?: string;
32
+ mode?: 'embedded' | 'floating';
33
+ floatingConfig?: {
34
+ position?: 'bottom-right' | 'bottom-left' | 'top-right' | 'top-left';
35
+ offset?: {
36
+ x?: number;
37
+ y?: number;
38
+ };
39
+ toggleButtonIcon?: React.ReactElement;
40
+ closeButtonIcon?: React.ReactElement;
41
+ };
42
+ }