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.
- package/README.md +113 -0
- package/assets/chat-logo.png +0 -0
- package/dist/components/ChatMessage.d.ts +20 -0
- package/dist/components/ChatMessage.js +188 -0
- package/dist/components/GenAgentChat.d.ts +3 -0
- package/dist/components/GenAgentChat.js +520 -0
- package/dist/components/VoiceInput.d.ts +14 -0
- package/dist/components/VoiceInput.js +51 -0
- package/dist/hooks/useChat.d.ts +21 -0
- package/dist/hooks/useChat.js +256 -0
- package/dist/hooks/useVoiceInput.d.ts +16 -0
- package/dist/hooks/useVoiceInput.js +210 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +2 -0
- package/dist/services/audioService.d.ts +17 -0
- package/dist/services/audioService.js +118 -0
- package/dist/services/chatService.d.ts +48 -0
- package/dist/services/chatService.js +376 -0
- package/dist/types/index.d.ts +42 -0
- package/dist/types/index.js +2 -0
- package/package.json +48 -0
|
@@ -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
|
+
}
|