@sinequa/assistant 3.6.2 → 3.7.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/chat/chat-message/chat-message.component.d.ts +4 -2
- package/chat/chat.component.d.ts +56 -17
- package/chat/chat.service.d.ts +23 -1
- package/chat/rest-chat.service.d.ts +1 -0
- package/chat/saved-chats/saved-chats.component.d.ts +2 -2
- package/chat/styles/assistant.scss +5 -0
- package/chat/types.d.ts +128 -3
- package/chat/websocket-chat.service.d.ts +6 -6
- package/esm2020/chat/chat-message/chat-message.component.mjs +9 -4
- package/esm2020/chat/chat-reference/chat-reference.component.mjs +3 -3
- package/esm2020/chat/chat.component.mjs +262 -60
- package/esm2020/chat/chat.service.mjs +49 -3
- package/esm2020/chat/debug-message/debug-message.component.mjs +3 -3
- package/esm2020/chat/rest-chat.service.mjs +8 -3
- package/esm2020/chat/saved-chats/saved-chats.component.mjs +16 -5
- package/esm2020/chat/token-progress-bar/token-progress-bar.component.mjs +3 -3
- package/esm2020/chat/types.mjs +11 -3
- package/esm2020/chat/websocket-chat.service.mjs +191 -99
- package/fesm2015/sinequa-assistant-chat.mjs +552 -178
- package/fesm2015/sinequa-assistant-chat.mjs.map +1 -1
- package/fesm2020/sinequa-assistant-chat.mjs +542 -175
- package/fesm2020/sinequa-assistant-chat.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -3,7 +3,7 @@ import { AuthenticationService } from "@sinequa/core/login";
|
|
|
3
3
|
import { SignalRWebService } from "@sinequa/core/web-services";
|
|
4
4
|
import { HttpTransportType, LogLevel } from "@microsoft/signalr";
|
|
5
5
|
import { ChatService } from "./chat.service";
|
|
6
|
-
import { merge, fromEvent, Observable, Subject, catchError, filter, forkJoin, map, shareReplay, switchMap, tap, throwError, takeUntil, take } from "rxjs";
|
|
6
|
+
import { merge, fromEvent, Observable, Subject, catchError, filter, forkJoin, map, shareReplay, switchMap, tap, throwError, takeUntil, take, mergeMap, of } from "rxjs";
|
|
7
7
|
import * as i0 from "@angular/core";
|
|
8
8
|
export class WebSocketChatService extends ChatService {
|
|
9
9
|
constructor() {
|
|
@@ -11,10 +11,8 @@ export class WebSocketChatService extends ChatService {
|
|
|
11
11
|
this._messageHandlers = new Map();
|
|
12
12
|
this._actionMap = new Map();
|
|
13
13
|
this._progress = undefined;
|
|
14
|
-
this._content = "";
|
|
15
14
|
this._attachments = [];
|
|
16
|
-
this.
|
|
17
|
-
this._debugMessages = undefined;
|
|
15
|
+
this._debugMessages = [];
|
|
18
16
|
this.signalRService = inject(SignalRWebService);
|
|
19
17
|
this.authenticationService = inject(AuthenticationService);
|
|
20
18
|
}
|
|
@@ -64,36 +62,36 @@ export class WebSocketChatService extends ChatService {
|
|
|
64
62
|
}
|
|
65
63
|
}
|
|
66
64
|
listModels() {
|
|
67
|
-
const modelsSubject = new Subject();
|
|
65
|
+
const modelsSubject$ = new Subject();
|
|
68
66
|
this.connection.on('ListModels', (res) => {
|
|
69
67
|
this.models = res.models?.filter(model => !!model.enable);
|
|
70
|
-
modelsSubject
|
|
71
|
-
modelsSubject
|
|
68
|
+
modelsSubject$.next(this.models);
|
|
69
|
+
modelsSubject$.complete();
|
|
72
70
|
});
|
|
73
71
|
// Send the request to get the list of models
|
|
74
72
|
this.connection.invoke('ListModels', { debug: this.chatConfig$.value.defaultValues.debug })
|
|
75
73
|
.catch(error => {
|
|
76
74
|
console.error('Error invoking ListModels:', error);
|
|
77
|
-
modelsSubject
|
|
78
|
-
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection
|
|
75
|
+
modelsSubject$.error(new Error(error));
|
|
76
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
|
|
79
77
|
});
|
|
80
|
-
return modelsSubject
|
|
78
|
+
return modelsSubject$.asObservable();
|
|
81
79
|
}
|
|
82
80
|
listFunctions() {
|
|
83
|
-
const functionsSubject = new Subject();
|
|
81
|
+
const functionsSubject$ = new Subject();
|
|
84
82
|
this.connection.on('ListFunctions', (res) => {
|
|
85
83
|
this.functions = res.functions?.filter(func => func.enabled);
|
|
86
|
-
functionsSubject
|
|
87
|
-
functionsSubject
|
|
84
|
+
functionsSubject$.next(this.functions);
|
|
85
|
+
functionsSubject$.complete();
|
|
88
86
|
});
|
|
89
87
|
// Send the request to get the list of functions
|
|
90
88
|
this.connection.invoke('ListFunctions', { debug: this.chatConfig$.value.defaultValues.debug })
|
|
91
89
|
.catch(error => {
|
|
92
90
|
console.error('Error invoking ListFunctions:', error);
|
|
93
|
-
functionsSubject
|
|
94
|
-
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection
|
|
91
|
+
functionsSubject$.error(new Error(error));
|
|
92
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
|
|
95
93
|
});
|
|
96
|
-
return functionsSubject
|
|
94
|
+
return functionsSubject$.asObservable();
|
|
97
95
|
}
|
|
98
96
|
fetch(messages, query) {
|
|
99
97
|
// Start streaming by invoking the Chat method
|
|
@@ -121,7 +119,8 @@ export class WebSocketChatService extends ChatService {
|
|
|
121
119
|
data.instanceId = this.chatInstanceId;
|
|
122
120
|
data.savedChatId = this.savedChatId;
|
|
123
121
|
}
|
|
124
|
-
|
|
122
|
+
// Initialize the response with an empty assistant message
|
|
123
|
+
this._response = [{ role: "assistant", content: "", additionalProperties: { display: true } }]; // here display: true is needed in order to be able to show the progress
|
|
125
124
|
// Create a Subject to signal completion
|
|
126
125
|
const completion$ = new Subject();
|
|
127
126
|
// Create observables for each non-global handler in the _messageHandlers map (default and eventual custom ones) once it is triggered by the hub connection
|
|
@@ -129,8 +128,19 @@ export class WebSocketChatService extends ChatService {
|
|
|
129
128
|
.from(this._messageHandlers.entries())
|
|
130
129
|
.filter(([eventName, eventHandler]) => !eventHandler.isGlobalHandler)
|
|
131
130
|
.map(([eventName, eventHandler]) => {
|
|
132
|
-
return fromEvent(this.connection, eventName).pipe(
|
|
133
|
-
|
|
131
|
+
return fromEvent(this.connection, eventName).pipe(mergeMap((event) => {
|
|
132
|
+
// Wrap the handler in a try-catch block to prevent the entire stream from failing if an error occurs in a single handler
|
|
133
|
+
try {
|
|
134
|
+
// Execute the handler and emit the result
|
|
135
|
+
// NB: here we could use [eventHandler.handler(event)] which behind the scenes mergeMap interprets this array as an observable sequence with one item, which it then emits
|
|
136
|
+
return of(eventHandler.handler(event));
|
|
137
|
+
}
|
|
138
|
+
catch (error) {
|
|
139
|
+
console.error(`Error in event handler for ${eventName}:`, error);
|
|
140
|
+
// Use throwError to propagate the error downstream
|
|
141
|
+
return throwError(() => new Error(`Error in event handler for ${eventName}: ${error}`));
|
|
142
|
+
}
|
|
143
|
+
}));
|
|
134
144
|
});
|
|
135
145
|
// Then merge them into a single observable in order to simulate the streaming behavior
|
|
136
146
|
const combined$ = merge(...observables).pipe(map(() => {
|
|
@@ -144,37 +154,74 @@ export class WebSocketChatService extends ChatService {
|
|
|
144
154
|
time: a.executionTime,
|
|
145
155
|
}))
|
|
146
156
|
: undefined;
|
|
147
|
-
//
|
|
148
|
-
|
|
149
|
-
|
|
157
|
+
// Always update ONLY the first assistant message of the _response with the new $progress, $attachment and $debug
|
|
158
|
+
// Assuming that the first assistant message is always visible since the hub does not send hidden messages by design
|
|
159
|
+
// So even if the first assistant message is hidden (display: false), the _response[0] will and should contain :
|
|
160
|
+
// - $progress, $attachment and $debug
|
|
161
|
+
// - the content of the first visible assistant message in the workflow
|
|
162
|
+
// This is mandatory in order to match the behavior of consecutive messages and maintain consistency with the chatHistory
|
|
163
|
+
if (!!this._progress || this._attachments.length > 0 || this._debugMessages.length > 0) {
|
|
164
|
+
this._response[0].additionalProperties.$progress = this._progress;
|
|
165
|
+
this._response[0].additionalProperties.$attachment = this._attachments;
|
|
166
|
+
this._response[0].additionalProperties.$debug = this._debugMessages;
|
|
150
167
|
}
|
|
151
168
|
// Return the result
|
|
152
|
-
return { history: [...messages,
|
|
169
|
+
return { history: [...messages, ...this._response], executionTime: this._executionTime };
|
|
153
170
|
}), takeUntil(completion$));
|
|
154
171
|
// return a new Observable that emits the result of the combined stream and handles the eventual errors of the invocation of the Chat method
|
|
155
172
|
return new Observable(observer => {
|
|
156
173
|
// Subscribe to combined stream
|
|
157
174
|
combined$.subscribe({
|
|
158
175
|
next: (value) => observer.next(value),
|
|
159
|
-
error: (err) => observer.error(err)
|
|
160
|
-
complete: () => observer.complete(),
|
|
176
|
+
error: (err) => observer.error(err)
|
|
161
177
|
});
|
|
162
178
|
// Invoke the Chat method and handle errors
|
|
163
179
|
this.connection.invoke('Chat', data)
|
|
180
|
+
.then(() => {
|
|
181
|
+
// If a valid assistant message with (display: true) was found, update it
|
|
182
|
+
// and it should always the case
|
|
183
|
+
const index = this.firstVisibleAssistantMessageIndex(this.chatHistory);
|
|
184
|
+
if (index !== -1) {
|
|
185
|
+
this.chatHistory[index].additionalProperties.$progress = this._progress;
|
|
186
|
+
this.chatHistory[index].additionalProperties.$attachment = this._attachments;
|
|
187
|
+
this.chatHistory[index].additionalProperties.$debug = this._debugMessages;
|
|
188
|
+
}
|
|
189
|
+
// Save/update the chat if savedChat enabled
|
|
190
|
+
if (this.chatConfig$.value.savedChatSettings.enabled && this.chatHistory.some((msg) => msg.additionalProperties?.isUserInput === true)) {
|
|
191
|
+
const action = !this.savedChatId ? this.addSavedChat(this.chatHistory).pipe(tap(() => this.listSavedChat())) : this.updateSavedChat(this.savedChatId, undefined, this.chatHistory);
|
|
192
|
+
action.pipe(take(1)).subscribe({
|
|
193
|
+
next: () => { },
|
|
194
|
+
error: (error) => {
|
|
195
|
+
this.streaming$.next(false);
|
|
196
|
+
observer.error(error);
|
|
197
|
+
},
|
|
198
|
+
complete: () => {
|
|
199
|
+
this.streaming$.next(false);
|
|
200
|
+
observer.complete();
|
|
201
|
+
}
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
else {
|
|
205
|
+
this.streaming$.next(false);
|
|
206
|
+
observer.complete();
|
|
207
|
+
}
|
|
208
|
+
})
|
|
164
209
|
.catch(error => {
|
|
165
210
|
console.error('Error invoking Chat:', error);
|
|
211
|
+
this.streaming$.next(false);
|
|
166
212
|
// Emit the error to the newly created observable
|
|
167
213
|
observer.error(error);
|
|
168
214
|
// Return a resolved promise to handle the error and prevent unhandled promise rejection
|
|
169
215
|
return Promise.resolve();
|
|
170
216
|
})
|
|
171
217
|
.finally(() => {
|
|
172
|
-
|
|
218
|
+
// This block concerns ONLY the completion of the "Chat" method invocation.
|
|
219
|
+
// This means the completion of the combined$ stream.
|
|
220
|
+
// It does not take into account the completion of the entire fetch method (the observable returned by fetch) and which depends on the completion of the save chat action if enabled
|
|
221
|
+
this._response = []; // Clear the _response
|
|
173
222
|
this._actionMap.clear(); // Clear the _actionMap
|
|
174
|
-
this._content = ""; // Clear the _content
|
|
175
223
|
this._progress = undefined; // Clear the _progress
|
|
176
224
|
this._attachments = []; // Clear the _attachments
|
|
177
|
-
this._suggestedActions = []; // Clear the _suggestedActions
|
|
178
225
|
this._debugMessages = []; // Clear the _debugMessages
|
|
179
226
|
this._executionTime = ""; // Clear the _executionTime
|
|
180
227
|
completion$.next(); // Emit a signal to complete the observables
|
|
@@ -182,6 +229,33 @@ export class WebSocketChatService extends ChatService {
|
|
|
182
229
|
});
|
|
183
230
|
});
|
|
184
231
|
}
|
|
232
|
+
stopGeneration() {
|
|
233
|
+
// Start stopping generation by invoking the CancelTasks method
|
|
234
|
+
this.stoppingGeneration$.next(true);
|
|
235
|
+
// Create a Subject to hold the result of the CancelTasks method
|
|
236
|
+
const stopGenerationSubject$ = new Subject();
|
|
237
|
+
this.connection.on('CancelTasks', (res) => {
|
|
238
|
+
// When the generation is stopped before streaming any VISIBLE assistant message, this means that $progress, $attachment and $debug properties will be lost.
|
|
239
|
+
// However, the "ContextMessage" frames will be persisted in the chatHistory and the assistant may reference them in the next generation.
|
|
240
|
+
// This leads to the problem of referencing undisplayed attachments in the next generation.
|
|
241
|
+
// To solve this problem, we need to persist $progress, $attachment and $debug properties by adding a new assistant message with empty content and these properties.
|
|
242
|
+
if (this._response.length === 1 && this._response[0].content === "") {
|
|
243
|
+
this.chatHistory?.push({ role: "assistant", content: "", additionalProperties: { display: true, $progress: this._progress, $attachment: this._attachments, $debug: this._debugMessages } });
|
|
244
|
+
}
|
|
245
|
+
stopGenerationSubject$.next(!!res); // Emit the result of the CancelTasks method
|
|
246
|
+
stopGenerationSubject$.complete(); // Complete the subject
|
|
247
|
+
this.stoppingGeneration$.next(false); // Complete stopping generation
|
|
248
|
+
});
|
|
249
|
+
// Invoke the CancelTasks method and handle errors
|
|
250
|
+
this.connection.invoke('CancelTasks')
|
|
251
|
+
.catch(error => {
|
|
252
|
+
console.error('Error invoking CancelTasks:', error);
|
|
253
|
+
stopGenerationSubject$.error(new Error(error));
|
|
254
|
+
this.stoppingGeneration$.next(false); // Complete stopping generation
|
|
255
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
|
|
256
|
+
});
|
|
257
|
+
return stopGenerationSubject$.asObservable();
|
|
258
|
+
}
|
|
185
259
|
listSavedChat() {
|
|
186
260
|
if (!this.chatConfig$.value.savedChatSettings.enabled) {
|
|
187
261
|
return;
|
|
@@ -201,27 +275,27 @@ export class WebSocketChatService extends ChatService {
|
|
|
201
275
|
});
|
|
202
276
|
}
|
|
203
277
|
getSavedChat(id) {
|
|
204
|
-
const savedChatSubject = new Subject();
|
|
278
|
+
const savedChatSubject$ = new Subject();
|
|
205
279
|
const data = {
|
|
206
280
|
instanceId: this.chatInstanceId,
|
|
207
281
|
savedChatId: id,
|
|
208
282
|
debug: this.chatConfig$.value.defaultValues.debug
|
|
209
283
|
};
|
|
210
284
|
this.connection.on('SavedChatGet', (res) => {
|
|
211
|
-
savedChatSubject
|
|
212
|
-
savedChatSubject
|
|
285
|
+
savedChatSubject$.next(res.savedChat);
|
|
286
|
+
savedChatSubject$.complete();
|
|
213
287
|
});
|
|
214
288
|
// Invoke the method SavedChatGet
|
|
215
289
|
this.connection.invoke('SavedChatGet', data)
|
|
216
290
|
.catch(error => {
|
|
217
291
|
console.error('Error invoking SavedChatGet:', error);
|
|
218
|
-
savedChatSubject
|
|
219
|
-
return Promise.resolve();
|
|
292
|
+
savedChatSubject$.error(new Error(error));
|
|
293
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
|
|
220
294
|
});
|
|
221
|
-
return savedChatSubject
|
|
295
|
+
return savedChatSubject$.asObservable();
|
|
222
296
|
}
|
|
223
297
|
addSavedChat(messages) {
|
|
224
|
-
const addSavedChatSubject = new Subject();
|
|
298
|
+
const addSavedChatSubject$ = new Subject();
|
|
225
299
|
const data = {
|
|
226
300
|
instanceId: this.chatInstanceId,
|
|
227
301
|
savedChatId: this.chatId,
|
|
@@ -231,20 +305,20 @@ export class WebSocketChatService extends ChatService {
|
|
|
231
305
|
this.connection.on('SavedChatAdd', (res) => {
|
|
232
306
|
this.setSavedChatId(res.savedChat.id); // Persist the savedChatId
|
|
233
307
|
this.generateAuditEvent('saved-chat.add', {}, res.savedChat.id); // Generate audit event
|
|
234
|
-
addSavedChatSubject
|
|
235
|
-
addSavedChatSubject
|
|
308
|
+
addSavedChatSubject$.next(res.savedChat);
|
|
309
|
+
addSavedChatSubject$.complete();
|
|
236
310
|
});
|
|
237
311
|
// Invoke the method SavedChatAdd
|
|
238
312
|
this.connection.invoke('SavedChatAdd', data)
|
|
239
313
|
.catch(error => {
|
|
240
314
|
console.error('Error invoking SavedChatAdd:', error);
|
|
241
|
-
addSavedChatSubject
|
|
242
|
-
return Promise.resolve();
|
|
315
|
+
addSavedChatSubject$.error(new Error(error));
|
|
316
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
|
|
243
317
|
});
|
|
244
|
-
return addSavedChatSubject
|
|
318
|
+
return addSavedChatSubject$.asObservable();
|
|
245
319
|
}
|
|
246
320
|
updateSavedChat(id, name, messages) {
|
|
247
|
-
const updateSavedChatSubject = new Subject();
|
|
321
|
+
const updateSavedChatSubject$ = new Subject();
|
|
248
322
|
const data = {
|
|
249
323
|
instanceId: this.chatInstanceId,
|
|
250
324
|
savedChatId: id,
|
|
@@ -255,45 +329,64 @@ export class WebSocketChatService extends ChatService {
|
|
|
255
329
|
if (messages)
|
|
256
330
|
data["history"] = messages;
|
|
257
331
|
this.connection.on('SavedChatUpdate', (res) => {
|
|
258
|
-
updateSavedChatSubject
|
|
259
|
-
updateSavedChatSubject
|
|
332
|
+
updateSavedChatSubject$.next(res.savedChat);
|
|
333
|
+
updateSavedChatSubject$.complete();
|
|
260
334
|
});
|
|
261
335
|
// Invoke the method SavedChatUpdate
|
|
262
336
|
this.connection.invoke('SavedChatUpdate', data)
|
|
263
337
|
.catch(error => {
|
|
264
338
|
console.error('Error invoking SavedChatUpdate:', error);
|
|
265
|
-
updateSavedChatSubject
|
|
266
|
-
return Promise.resolve();
|
|
339
|
+
updateSavedChatSubject$.error(new Error(error));
|
|
340
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
|
|
267
341
|
});
|
|
268
|
-
return updateSavedChatSubject
|
|
342
|
+
return updateSavedChatSubject$.asObservable();
|
|
269
343
|
}
|
|
270
344
|
deleteSavedChat(ids) {
|
|
271
|
-
const deleteSavedChatSubject = new Subject();
|
|
345
|
+
const deleteSavedChatSubject$ = new Subject();
|
|
272
346
|
const data = {
|
|
273
347
|
instanceId: this.chatInstanceId,
|
|
274
348
|
SavedChatIds: ids,
|
|
275
349
|
debug: this.chatConfig$.value.defaultValues.debug
|
|
276
350
|
};
|
|
277
351
|
this.connection.on('SavedChatDelete', (res) => {
|
|
278
|
-
deleteSavedChatSubject
|
|
279
|
-
deleteSavedChatSubject
|
|
352
|
+
deleteSavedChatSubject$.next(res.deleteCount);
|
|
353
|
+
deleteSavedChatSubject$.complete();
|
|
280
354
|
});
|
|
281
355
|
// Invoke the method SavedChatDelete
|
|
282
356
|
this.connection.invoke('SavedChatDelete', data)
|
|
283
357
|
.catch(error => {
|
|
284
358
|
console.error('Error invoking SavedChatDelete:', error);
|
|
285
|
-
deleteSavedChatSubject
|
|
286
|
-
return Promise.resolve();
|
|
359
|
+
deleteSavedChatSubject$.error(new Error(error));
|
|
360
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
|
|
287
361
|
});
|
|
288
|
-
return deleteSavedChatSubject
|
|
362
|
+
return deleteSavedChatSubject$.asObservable();
|
|
289
363
|
}
|
|
290
364
|
/**
|
|
291
365
|
* Initialize out-of-the-box handlers
|
|
292
|
-
* It is a placeholder for non-streaming scenarios, where you invoke a specific hub method, and the server responds with
|
|
366
|
+
* It is a placeholder for non-streaming scenarios, where you invoke a specific hub method, and the server responds with frame message(s)
|
|
293
367
|
*/
|
|
294
368
|
initMessageHandlers() {
|
|
295
|
-
this.addMessageHandler("
|
|
296
|
-
|
|
369
|
+
this.addMessageHandler("Error", {
|
|
370
|
+
handler: (error) => {
|
|
371
|
+
console.error(error);
|
|
372
|
+
this.notificationsService.error(error);
|
|
373
|
+
},
|
|
374
|
+
isGlobalHandler: true
|
|
375
|
+
});
|
|
376
|
+
this.addMessageHandler("Quota", {
|
|
377
|
+
handler: (message) => {
|
|
378
|
+
try {
|
|
379
|
+
this.updateQuota(message.quota);
|
|
380
|
+
}
|
|
381
|
+
catch (error) {
|
|
382
|
+
console.error(error);
|
|
383
|
+
}
|
|
384
|
+
},
|
|
385
|
+
isGlobalHandler: true
|
|
386
|
+
});
|
|
387
|
+
this.addMessageHandler("Debug", { handler: () => { },
|
|
388
|
+
isGlobalHandler: true
|
|
389
|
+
});
|
|
297
390
|
this.addMessageHandler("ActionStart", { handler: (action) => this._actionMap.set(action.guid, action),
|
|
298
391
|
isGlobalHandler: false });
|
|
299
392
|
this.addMessageHandler("ActionResult", {
|
|
@@ -309,35 +402,48 @@ export class WebSocketChatService extends ChatService {
|
|
|
309
402
|
isGlobalHandler: false
|
|
310
403
|
});
|
|
311
404
|
this.addMessageHandler("Message", {
|
|
312
|
-
handler: (message) => this.
|
|
405
|
+
handler: (message) => this._response.at(-1).content += message ?? "",
|
|
313
406
|
isGlobalHandler: false
|
|
314
407
|
});
|
|
315
408
|
this.addMessageHandler("History", {
|
|
316
409
|
handler: (history) => {
|
|
317
|
-
|
|
318
|
-
//
|
|
319
|
-
this.chatHistory
|
|
320
|
-
|
|
321
|
-
this.chatHistory.at(-1).additionalProperties.$suggestedAction = this._suggestedActions;
|
|
322
|
-
this.chatHistory.at(-1).additionalProperties.$debug = this._debugMessages;
|
|
323
|
-
// Emit the updated chat usage metrics once the generation of the assistant response is completed
|
|
410
|
+
// The ChatHistory is updated: it is the current copy concatenated with the new items ONLY (it can have multiple messages: the context messages + the response message)
|
|
411
|
+
// This is mandatory to not lose the previous updates of the chatHistory when the assistant is streaming multiple message steps
|
|
412
|
+
this.chatHistory = [...this.chatHistory, ...(history.history.slice(this.chatHistory.length))];
|
|
413
|
+
// Emit the updated chat usage metrics
|
|
324
414
|
if (!!this.chatHistory.at(-1)?.additionalProperties.usageMetrics) {
|
|
325
415
|
this.updateChatUsageMetrics(this.chatHistory.at(-1).additionalProperties.usageMetrics);
|
|
326
416
|
}
|
|
327
|
-
// Save/update the chat if savedChat enabled
|
|
328
|
-
if (this.chatConfig$.value.savedChatSettings.enabled && this.chatHistory.some((msg) => msg.additionalProperties?.isUserInput === true)) {
|
|
329
|
-
const action = !this.savedChatId ? this.addSavedChat(this.chatHistory) : this.updateSavedChat(this.savedChatId, undefined, this.chatHistory);
|
|
330
|
-
action.pipe(take(1)).subscribe();
|
|
331
|
-
}
|
|
332
417
|
this._executionTime = history.executionTime;
|
|
418
|
+
},
|
|
419
|
+
isGlobalHandler: false
|
|
420
|
+
});
|
|
421
|
+
this.addMessageHandler("SuggestedActions", {
|
|
422
|
+
handler: (message) => {
|
|
423
|
+
// Since after the "History" and "MessageBreak" that this event is caught,
|
|
424
|
+
// $suggestedAction needs to be updated directly to the last visible "assistant" message in the _response and the chatHistory
|
|
425
|
+
this._response.at(-1).additionalProperties.$suggestedAction = (this._response.at(-1).additionalProperties.$suggestedAction || []).concat(message.suggestedActions);
|
|
426
|
+
const index = this.lastVisibleAssistantMessageIndex(this.chatHistory);
|
|
427
|
+
if (index !== -1) {
|
|
428
|
+
this.chatHistory[index].additionalProperties.$suggestedAction = (this.chatHistory[index].additionalProperties.$suggestedAction || []).concat(message.suggestedActions);
|
|
429
|
+
}
|
|
430
|
+
},
|
|
431
|
+
isGlobalHandler: false
|
|
432
|
+
});
|
|
433
|
+
this.addMessageHandler("DebugDisplay", {
|
|
434
|
+
handler: (message) => this._debugMessages = this._debugMessages.concat(message),
|
|
435
|
+
isGlobalHandler: false
|
|
436
|
+
});
|
|
437
|
+
this.addMessageHandler("MessageBreak", {
|
|
438
|
+
handler: () => {
|
|
333
439
|
// Generate audit event
|
|
334
440
|
const details = {
|
|
335
441
|
'duration': this._executionTime,
|
|
336
442
|
'text': this.chatHistory.at(-1).content,
|
|
337
443
|
'role': this.chatHistory.at(-1).role,
|
|
338
444
|
'rank': this.chatHistory.length - 1,
|
|
339
|
-
'generation-tokencount': this.chatHistory.at(-1).additionalProperties.usageMetrics
|
|
340
|
-
'prompt-tokencount': this.chatHistory.at(-1).additionalProperties.usageMetrics
|
|
445
|
+
'generation-tokencount': this.chatHistory.at(-1).additionalProperties.usageMetrics?.completionTokenCount,
|
|
446
|
+
'prompt-tokencount': this.chatHistory.at(-1).additionalProperties.usageMetrics?.promptTokenCount,
|
|
341
447
|
'attachments': JSON.stringify(this._attachments.map(({ recordId, contextId, parts, type }) => ({
|
|
342
448
|
recordId,
|
|
343
449
|
contextId,
|
|
@@ -346,31 +452,13 @@ export class WebSocketChatService extends ChatService {
|
|
|
346
452
|
})))
|
|
347
453
|
};
|
|
348
454
|
this.generateAuditEvent('message', details);
|
|
455
|
+
// Push a new assistant message to the _response array ONLY if the content of the last message is not empty
|
|
456
|
+
if (this._response.at(-1).content !== "") {
|
|
457
|
+
this._response.push({ role: "assistant", content: "", additionalProperties: { display: true } });
|
|
458
|
+
}
|
|
349
459
|
},
|
|
350
460
|
isGlobalHandler: false
|
|
351
461
|
});
|
|
352
|
-
this.addMessageHandler("Error", {
|
|
353
|
-
handler: (error) => {
|
|
354
|
-
console.error(error);
|
|
355
|
-
this.notificationsService.error(error);
|
|
356
|
-
},
|
|
357
|
-
isGlobalHandler: true
|
|
358
|
-
});
|
|
359
|
-
this.addMessageHandler("Quota", {
|
|
360
|
-
handler: (message) => this.updateQuota(message.quota),
|
|
361
|
-
isGlobalHandler: true
|
|
362
|
-
});
|
|
363
|
-
this.addMessageHandler("SuggestedActions", {
|
|
364
|
-
handler: (message) => {
|
|
365
|
-
// Suggested actions are concatenated to the existing ones in case of multiple events "SuggestedActions"
|
|
366
|
-
this._suggestedActions = this._suggestedActions.concat(message.suggestedActions);
|
|
367
|
-
},
|
|
368
|
-
isGlobalHandler: false
|
|
369
|
-
});
|
|
370
|
-
this.addMessageHandler("DebugDisplay", {
|
|
371
|
-
handler: (message) => this._debugMessages = this._debugMessages ? this._debugMessages.concat(message) : [message],
|
|
372
|
-
isGlobalHandler: false
|
|
373
|
-
});
|
|
374
462
|
}
|
|
375
463
|
/**
|
|
376
464
|
* Override and register the entire _messageHandlers map by merging the provided map with the default one
|
|
@@ -385,7 +473,7 @@ export class WebSocketChatService extends ChatService {
|
|
|
385
473
|
});
|
|
386
474
|
// Merge the new event handlers with the existing ones
|
|
387
475
|
this._messageHandlers = new Map([...this._messageHandlers, ..._messageHandlers]);
|
|
388
|
-
// Register the global
|
|
476
|
+
// Register the global handlers among the merged map
|
|
389
477
|
this._messageHandlers.forEach((eventHandler, eventName) => {
|
|
390
478
|
if (eventHandler.isGlobalHandler) {
|
|
391
479
|
this.registerMessageHandler(eventName, eventHandler);
|
|
@@ -395,7 +483,7 @@ export class WebSocketChatService extends ChatService {
|
|
|
395
483
|
/**
|
|
396
484
|
* Add a listener for a specific event.
|
|
397
485
|
* If a listener for this same event already exists, it will be overridden.
|
|
398
|
-
* If the listener has "
|
|
486
|
+
* If the listener has "isGlobalHandler" set to true, it will be registered to the hub connection.
|
|
399
487
|
* @param eventName Name of the event to register a listener for
|
|
400
488
|
* @param eventHandler The handler to be called when the event is received
|
|
401
489
|
*/
|
|
@@ -449,8 +537,12 @@ export class WebSocketChatService extends ChatService {
|
|
|
449
537
|
reject(new Error("No endpoint provided to connect the websocket to"));
|
|
450
538
|
return;
|
|
451
539
|
}
|
|
452
|
-
const logLevel = this.
|
|
540
|
+
const logLevel = this._getLogLevel();
|
|
453
541
|
this.connection = this.signalRService.buildConnection(this.REQUEST_URL, { ...this.defaultOptions, ...options }, logLevel, true);
|
|
542
|
+
const signalRServerTimeoutInMilliseconds = this.chatConfig$.value?.connectionSettings.signalRServerTimeoutInMilliseconds;
|
|
543
|
+
if (signalRServerTimeoutInMilliseconds) {
|
|
544
|
+
this.connection.serverTimeoutInMilliseconds = signalRServerTimeoutInMilliseconds;
|
|
545
|
+
}
|
|
454
546
|
resolve();
|
|
455
547
|
});
|
|
456
548
|
}
|
|
@@ -468,7 +560,7 @@ export class WebSocketChatService extends ChatService {
|
|
|
468
560
|
stopConnection() {
|
|
469
561
|
return this.signalRService.stopConnection(this.connection);
|
|
470
562
|
}
|
|
471
|
-
|
|
563
|
+
_getTransports() {
|
|
472
564
|
switch (this.chatConfig$.value?.connectionSettings.signalRTransport) {
|
|
473
565
|
case "WebSockets":
|
|
474
566
|
return HttpTransportType.WebSockets;
|
|
@@ -480,7 +572,7 @@ export class WebSocketChatService extends ChatService {
|
|
|
480
572
|
return HttpTransportType.None;
|
|
481
573
|
}
|
|
482
574
|
}
|
|
483
|
-
|
|
575
|
+
_getLogLevel() {
|
|
484
576
|
switch (this.chatConfig$.value?.connectionSettings.signalRLogLevel) {
|
|
485
577
|
case "Critical":
|
|
486
578
|
return LogLevel.Critical; // Log level for diagnostic messages that indicate a failure that will terminate the entire application.
|
|
@@ -514,7 +606,7 @@ export class WebSocketChatService extends ChatService {
|
|
|
514
606
|
// as far as we know, signalR only lets us tweak the request with this access token factory
|
|
515
607
|
// so we pass along the Sinequa CSRF token to pass the CSRF check..
|
|
516
608
|
return {
|
|
517
|
-
transport: this.
|
|
609
|
+
transport: this._getTransports(),
|
|
518
610
|
withCredentials: true,
|
|
519
611
|
headers,
|
|
520
612
|
accessTokenFactory: () => this.authenticationService.processedCredentials?.data?.csrfToken || ""
|
|
@@ -526,4 +618,4 @@ WebSocketChatService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0",
|
|
|
526
618
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: WebSocketChatService, decorators: [{
|
|
527
619
|
type: Injectable
|
|
528
620
|
}], ctorParameters: function () { return []; } });
|
|
529
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-chat.service.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/websocket-chat.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from \"@angular/core\";\nimport { AuthenticationService } from \"@sinequa/core/login\";\nimport { ConnectionOptions, SignalRWebService } from \"@sinequa/core/web-services\";\nimport { Query } from \"@sinequa/core/app-utils\";\nimport { HttpTransportType, HubConnection, LogLevel, MessageHeaders } from \"@microsoft/signalr\";\nimport { ActionMessage, ActionResultEvent, ActionStartEvent, ActionStopEvent, MessageEvent, ChatMessage, ChatPayload, ChatResponse, GllmFunction, GllmModelDescription, MessageHandler, HistoryEvent, ErrorEvent, QuotaEvent, ChatContextAttachment, ContextMessageEvent, SavedChatHistory, SavedChat, ChatProgress, SuggestedActionsEvent, SuggestedAction, DebugMessageEvent, DebugMessage } from \"./types\";\nimport { ChatService } from \"./chat.service\";\nimport { merge, fromEvent, Observable, Subject, catchError, filter, forkJoin, map, shareReplay, switchMap, tap, throwError, takeUntil, take } from \"rxjs\";\n\n@Injectable()\nexport class WebSocketChatService extends ChatService {\n\n  public connection: HubConnection | undefined;\n\n  private _messageHandlers: Map<string, MessageHandler<any>> = new Map();\n  private _actionMap = new Map<string, ActionMessage>();\n  private _progress: ChatProgress[] | undefined = undefined;\n  private _content = \"\";\n  private _executionTime: string;\n  private _attachments: ChatContextAttachment[] = [];\n  private _suggestedActions: SuggestedAction[] = [];\n  private _debugMessages: DebugMessage[] | undefined = undefined;\n\n  public signalRService = inject(SignalRWebService);\n  public authenticationService = inject(AuthenticationService);\n\n  constructor() {\n    super();\n  }\n\n  /**\n   * Initialize the chat process after the login is complete.\n   * It includes building and starting a connection, executing parallel requests for models and functions, and handling errors during the process.\n   *\n   * @returns An Observable<boolean> indicating the success of the initialization process.\n   */\n  init(): Observable<boolean> {\n    return this.loginService.events.pipe(\n      filter((e) => e.type === 'login-complete'),\n      tap(() => this.getRequestsUrl()),\n      // Build the connection\n      switchMap(() => this.buildConnection()),\n      tap(() => this.initMessageHandlers()),\n      // Start the connection\n      switchMap(() => this.startConnection()),\n      // Execute parallel requests for models and functions\n      switchMap(() => {\n        return forkJoin([\n          this.listModels(),\n          this.listFunctions()\n        ])\n      }),\n      // Map the results of parallel requests to a boolean indicating success\n      map(([models, functions]) => {\n        const result = !!models && !!functions;\n        this.initProcess$.next(result);\n        return result;\n      }),\n      // Any errors during the process are caught, logged, and re-thrown to propagate the error further\n      catchError((error) => {\n        console.error('Error occurred:', error);\n        return throwError(() => error);\n      }),\n      // Cache and replay the emitted value for subsequent subscribers\n      shareReplay(1)\n    );\n  }\n\n  /**\n   * Define the assistant endpoint to use for the websocket requests\n   * It can be overridden by the app config\n   */\n  getRequestsUrl() {\n    if (this.chatConfig$.value!.connectionSettings.websocketEndpoint) {\n      this.REQUEST_URL = this.chatConfig$.value!.connectionSettings.websocketEndpoint;\n    } else {\n      throw new Error(`The property 'websocketEndpoint' must be provided when attempting to use 'WebSocket' in assistant instance`);\n    }\n  }\n\n  listModels(): Observable<GllmModelDescription[] | undefined> {\n    const modelsSubject = new Subject<GllmModelDescription[] | undefined>();\n\n    this.connection!.on('ListModels', (res) => {\n      this.models = (res.models as GllmModelDescription[] | undefined)?.filter(model => !!model.enable);\n      modelsSubject.next(this.models);\n      modelsSubject.complete()\n    });\n\n    // Send the request to get the list of models\n    this.connection!.invoke('ListModels', { debug: this.chatConfig$.value!.defaultValues.debug })\n      .catch(error => {\n        console.error('Error invoking ListModels:', error);\n        modelsSubject.complete()\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection\n      })\n\n    return modelsSubject.asObservable();\n  }\n\n  listFunctions(): Observable<GllmFunction[] | undefined> {\n    const functionsSubject = new Subject<GllmFunction[] | undefined>();\n\n    this.connection!.on('ListFunctions', (res) => {\n      this.functions = (res.functions as GllmFunction[] | undefined)?.filter(func => func.enabled);\n      functionsSubject.next(this.functions);\n      functionsSubject.complete()\n    });\n\n    // Send the request to get the list of functions\n    this.connection!.invoke('ListFunctions', { debug: this.chatConfig$.value!.defaultValues.debug })\n      .catch(error => {\n        console.error('Error invoking ListFunctions:', error);\n        functionsSubject.complete()\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection\n      })\n\n    return functionsSubject.asObservable();\n  }\n\n  fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse> {\n    // Start streaming by invoking the Chat method\n    this.streaming$.next(true);\n\n    // Prepare the payload to send to the Chat method\n    const data: ChatPayload = {\n      history: messages,\n      functions: this.chatConfig$.value!.defaultValues.functions?.filter(func => func.enabled).map(func => func.name),\n      debug: this.chatConfig$.value!.defaultValues.debug,\n      serviceSettings: {\n        service_id: this.chatConfig$.value!.defaultValues.service_id,\n        model_id: this.chatConfig$.value!.defaultValues.model_id,\n        top_p: this.chatConfig$.value!.defaultValues.top_p,\n        temperature: this.chatConfig$.value!.defaultValues.temperature,\n        max_tokens: this.chatConfig$.value!.defaultValues.max_tokens,\n        ...this.chatConfig$.value!.additionalServiceSettings\n      },\n      appQuery: {\n        app: this.appService.appName,\n        query\n      },\n      genericChatErrorMessage: this.chatConfig$.value!.globalSettings.genericChatErrorMessage\n    }\n    if (this.chatConfig$.value!.savedChatSettings.enabled) {\n      data.instanceId = this.chatInstanceId;\n      data.savedChatId = this.savedChatId;\n    }\n\n    let response: ChatMessage = {role: \"assistant\", content: \"\", additionalProperties: {display: true}}; // here display: true is needed in order to be able to show the progress\n\n    // Create a Subject to signal completion\n    const completion$ = new Subject<void>();\n\n    // Create observables for each non-global handler in the _messageHandlers map (default and eventual custom ones) once it is triggered by the hub connection\n    const observables = Array\n      .from(this._messageHandlers.entries())\n      .filter(([eventName, eventHandler]) => !eventHandler.isGlobalHandler)\n      .map(([eventName, eventHandler]) => {\n        return fromEvent<any>(this.connection!, eventName).pipe(\n          map((event) => eventHandler.handler(event)) // Execute the corresponding handler\n        );\n      });\n\n    // Then merge them into a single observable in order to simulate the streaming behavior\n    const combined$ = merge(...observables).pipe(\n      map(() => {\n        // Define $progress from the _actionMap\n        const actions = Array.from(this._actionMap.values());\n        this._progress = actions.length > 0\n                          ? actions.map((a) => ({\n                                title: a.displayName ?? \"\",\n                                content: a.displayValue ?? \"\",\n                                done: a.executionTime !== undefined,\n                                time: a.executionTime,\n                              }))\n                          : undefined;\n\n        // As soon as the first _content, $progress, _attachments or __suggestedActions is defined and not empty, the assistant is considered as streaming\n        if(!!this._content || this._progress || this._attachments.length > 0 || this._suggestedActions.length > 0) {\n          response = {...response, content: this._content, additionalProperties: {...response.additionalProperties, $progress: this._progress, $attachment: this._attachments, $suggestedAction: this._suggestedActions, $debug: this._debugMessages}};\n        }\n\n        // Return the result\n        return { history: [...messages, response], executionTime: this._executionTime };\n      }),\n      takeUntil(completion$), // Complete the observable when completion$ emits\n    );\n\n    // return a new Observable that emits the result of the combined stream and handles the eventual errors of the invocation of the Chat method\n    return new Observable(observer => {\n      // Subscribe to combined stream\n      combined$.subscribe({\n        next: (value) => observer.next(value),\n        error: (err) => observer.error(err),\n        complete: () => observer.complete(),\n      });\n\n      // Invoke the Chat method and handle errors\n      this.connection!.invoke('Chat', data)\n        .catch(error => {\n          console.error('Error invoking Chat:', error);\n          // Emit the error to the newly created observable\n          observer.error(error);\n          // Return a resolved promise to handle the error and prevent unhandled promise rejection\n          return Promise.resolve();\n        })\n        .finally(() => {\n          this.streaming$.next(false); // Complete streaming regardless of success or error\n          this._actionMap.clear(); // Clear the _actionMap\n          this._content = \"\"; // Clear the _content\n          this._progress = undefined; // Clear the _progress\n          this._attachments = []; // Clear the _attachments\n          this._suggestedActions = []; // Clear the _suggestedActions\n          this._debugMessages = []; // Clear the _debugMessages\n          this._executionTime = \"\"; // Clear the _executionTime\n          completion$.next(); // Emit a signal to complete the observables\n          completion$.complete(); // Complete the subject\n        });\n    });\n  }\n\n  listSavedChat(): void {\n    if (!this.chatConfig$.value!.savedChatSettings.enabled) {\n      return;\n    }\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatList', (res) => {\n      this.savedChats$.next(res.savedChats); // emits the result to the savedChats$ subject\n    });\n\n    // Invoke the method SavedChatList\n    this.connection!.invoke('SavedChatList', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatList:', error);\n        return Promise.resolve();\n      });\n  }\n\n  getSavedChat(id: string): Observable<SavedChatHistory | undefined> {\n    const savedChatSubject = new Subject<SavedChatHistory | undefined>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatGet', (res) => {\n      savedChatSubject.next(res.savedChat);\n      savedChatSubject.complete()\n    });\n\n    // Invoke the method SavedChatGet\n    this.connection!.invoke('SavedChatGet', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatGet:', error);\n        savedChatSubject.complete()\n        return Promise.resolve();\n      })\n\n    return savedChatSubject.asObservable();\n  }\n\n  addSavedChat(messages: ChatMessage[]): Observable<SavedChat> {\n    const addSavedChatSubject = new Subject<SavedChat>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      savedChatId: this.chatId,\n      history: messages,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatAdd', (res) => {\n      this.setSavedChatId(res.savedChat.id); // Persist the savedChatId\n      this.generateAuditEvent('saved-chat.add', {}, res.savedChat.id); // Generate audit event\n      addSavedChatSubject.next(res.savedChat);\n      addSavedChatSubject.complete()\n    });\n\n    // Invoke the method SavedChatAdd\n    this.connection!.invoke('SavedChatAdd', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatAdd:', error);\n        addSavedChatSubject.complete()\n        return Promise.resolve();\n      })\n\n    return addSavedChatSubject.asObservable();\n  }\n\n  updateSavedChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChat> {\n    const updateSavedChatSubject = new Subject<SavedChat>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    if(name) data[\"title\"] = name;\n    if(messages) data[\"history\"] = messages;\n\n    this.connection!.on('SavedChatUpdate', (res) => {\n      updateSavedChatSubject.next(res.savedChat);\n      updateSavedChatSubject.complete()\n    });\n\n    // Invoke the method SavedChatUpdate\n    this.connection!.invoke('SavedChatUpdate', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatUpdate:', error);\n        updateSavedChatSubject.complete()\n        return Promise.resolve();\n      })\n\n    return updateSavedChatSubject.asObservable();\n  }\n\n  deleteSavedChat(ids: string[]): Observable<number> {\n    const deleteSavedChatSubject = new Subject<number>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      SavedChatIds: ids,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatDelete', (res) => {\n      deleteSavedChatSubject.next(res.deleteCount);\n      deleteSavedChatSubject.complete();\n    });\n\n    // Invoke the method SavedChatDelete\n    this.connection!.invoke('SavedChatDelete', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatDelete:', error);\n        deleteSavedChatSubject.complete();\n        return Promise.resolve();\n      })\n\n    return deleteSavedChatSubject.asObservable();\n  }\n\n  /**\n   * Initialize out-of-the-box handlers\n   * It is a placeholder for non-streaming scenarios, where you invoke a specific hub method, and the server responds with a single message or a result\n   */\n  initMessageHandlers() {\n    this.addMessageHandler(\n      \"Debug\",\n      { handler: (debug: any) => console.log(debug),\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"ActionStart\",\n      { handler: (action: ActionStartEvent) => this._actionMap.set(action.guid, action),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ActionResult\",\n      {\n        handler: (action: ActionResultEvent) => this._actionMap.set(action.guid, { ...this._actionMap.get(action.guid), ...action }),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ActionStop\",\n      {\n        handler: (action: ActionStopEvent) => this._actionMap.set(action.guid, { ...this._actionMap.get(action.guid), ...action }),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ContextMessage\",\n      {\n        handler: (message: ContextMessageEvent) => this._attachments.push(message.metadata),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"Message\",\n      {\n        handler: (message: MessageEvent) => this._content += message ?? \"\",\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"History\",\n      {\n        handler: (history: HistoryEvent) => {\n          this.chatHistory = history.history;\n          // Re-attach the {$progress, $attachment, $suggestedAction and $debug} of the last response to the last assistant's response in the chat history\n          this.chatHistory.at(-1)!.additionalProperties.$progress = this._progress;\n          this.chatHistory.at(-1)!.additionalProperties.$attachment = this._attachments;\n          this.chatHistory.at(-1)!.additionalProperties.$suggestedAction = this._suggestedActions;\n          this.chatHistory.at(-1)!.additionalProperties.$debug = this._debugMessages;\n          // Emit the updated chat usage metrics once the generation of the assistant response is completed\n          if (!!this.chatHistory.at(-1)?.additionalProperties.usageMetrics) {\n            this.updateChatUsageMetrics(this.chatHistory.at(-1)!.additionalProperties.usageMetrics!);\n          }\n          // Save/update the chat if savedChat enabled\n          if (this.chatConfig$.value!.savedChatSettings.enabled && this.chatHistory.some((msg) => msg.additionalProperties?.isUserInput === true)) {\n            const action = !this.savedChatId ? this.addSavedChat(this.chatHistory) : this.updateSavedChat(this.savedChatId, undefined, this.chatHistory);\n            action.pipe(take(1)).subscribe();\n          }\n          this._executionTime = history.executionTime;\n          // Generate audit event\n          const details = {\n            'duration': this._executionTime,\n            'text': this.chatHistory.at(-1)!.content,\n            'role': this.chatHistory.at(-1)!.role, // 'assistant'\n            'rank': this.chatHistory.length - 1,\n            'generation-tokencount': this.chatHistory.at(-1)!.additionalProperties.usageMetrics!.completionTokenCount,\n            'prompt-tokencount': this.chatHistory.at(-1)!.additionalProperties.usageMetrics!.promptTokenCount,\n            'attachments': JSON.stringify(this._attachments.map(({ recordId, contextId, parts, type }) => ({\n                            recordId,\n                            contextId,\n                            parts: parts.map(({ partId, text }) => ({ partId, text })),\n                            type\n                          })))\n          };\n          this.generateAuditEvent('message', details);\n        },\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"Error\",\n      {\n        handler: (error: ErrorEvent) => {\n          console.error(error);\n          this.notificationsService.error(error);\n        },\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"Quota\",\n      {\n        handler: (message: QuotaEvent) => this.updateQuota(message.quota),\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"SuggestedActions\",\n      {\n        handler: (message: SuggestedActionsEvent) => {\n          // Suggested actions are concatenated to the existing ones in case of multiple events \"SuggestedActions\"\n          this._suggestedActions = this._suggestedActions.concat(message.suggestedActions);\n        },\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"DebugDisplay\",\n      {\n        handler: (message: DebugMessageEvent) => this._debugMessages = this._debugMessages ? this._debugMessages.concat(message) : [message],\n        isGlobalHandler: false\n      }\n    );\n  }\n\n  /**\n   * Override and register the entire _messageHandlers map by merging the provided map with the default one\n   * @param _messageHandlers\n   */\n  overrideMessageHandlers<T>(_messageHandlers: Map<string, MessageHandler<T>>) {\n    // Clear the already registered global chat handlers before merging the new ones\n    this._messageHandlers.forEach((eventHandler, eventName) => {\n      if(eventHandler.isGlobalHandler) {\n        this.unsubscribeMessageHandler(eventName);\n      }\n    });\n\n    // Merge the new event handlers with the existing ones\n    this._messageHandlers = new Map([...this._messageHandlers, ..._messageHandlers]);\n\n    // Register the global chat handlers among the merged map\n    this._messageHandlers.forEach((eventHandler, eventName) => {\n      if(eventHandler.isGlobalHandler) {\n        this.registerMessageHandler(eventName, eventHandler);\n      }\n    });\n  }\n\n  /**\n   * Add a listener for a specific event.\n   * If a listener for this same event already exists, it will be overridden.\n   * If the listener has \"isChatGlobalHandler\" set to true, it will be registered to the hub connection.\n   * @param eventName Name of the event to register a listener for\n   * @param eventHandler The handler to be called when the event is received\n   */\n  addMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>) {\n    this._messageHandlers.set(eventName, eventHandler);\n    if(eventHandler.isGlobalHandler) {\n      this.registerMessageHandler(eventName, eventHandler);\n    }\n  }\n\n  /**\n   * Dynamically register a listener for a specific event.\n   * If a listener for this event already exists, it will be overridden.\n   * @param eventName Name of the event to register a listener for\n   * @param eventHandler The handler to be called when the event is received\n   */\n  protected registerMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>) {\n    if (!this.connection) {\n      console.log(\"No connection found to register the listener\" + eventName);\n      return;\n    }\n\n    this.connection.on(eventName, (data: T) => {\n      eventHandler.handler(data);\n    });\n  }\n\n  /**\n   * Remove a listener for a specific event from the _messageHandlers map and unsubscribe from receiving messages for this event from the SignalR hub.\n   * @param eventName Name of the event to remove the listener for\n   */\n  removeMessageHandler(eventName: string) {\n    this._messageHandlers.delete(eventName);\n    this.unsubscribeMessageHandler(eventName);\n  }\n\n  /**\n   * Unsubscribe from receiving messages for a specific event from the SignalR hub.\n   * ALL its related listeners will be removed from hub connection\n   * This is needed to prevent accumulating old listeners when overriding the entire _messageHandlers map\n   * @param eventName Name of the event\n   */\n  protected unsubscribeMessageHandler(eventName: string) {\n    this.connection!.off(eventName);\n  }\n\n  /**\n   * Build a connection to the signalR websocket and register default listeners to the methods defined in the server hub class\n   * @param options The options for the connection. It overrides the default options\n   * @param logLevel Define the log level displayed in the console\n   * @returns Promise that resolves when the connection is built\n   */\n  buildConnection(options?: ConnectionOptions): Promise<void> {\n    return new Promise<void>((resolve, reject) => {\n      if (!this.REQUEST_URL) {\n          reject(new Error(\"No endpoint provided to connect the websocket to\"));\n          return;\n      }\n      const logLevel = this.getLogLevel();\n      this.connection = this.signalRService.buildConnection(this.REQUEST_URL, {...this.defaultOptions, ...options}, logLevel, true);\n      resolve();\n    });\n  }\n\n  /**\n   * Start the connection\n   * @returns Promise that resolves when the connection is started\n   */\n  startConnection(): Promise<void> {\n    return this.signalRService.startConnection(this.connection);\n  }\n\n  /**\n   * Stop the connection\n   * @returns Promise that resolves when the connection is stopped\n   */\n  stopConnection(): Promise<void> {\n    return this.signalRService.stopConnection(this.connection);\n  }\n\n  private getTransports(): HttpTransportType {\n    switch (this.chatConfig$.value?.connectionSettings.signalRTransport) {\n      case \"WebSockets\":\n        return HttpTransportType.WebSockets;\n      case \"ServerSentEvents\":\n        return HttpTransportType.ServerSentEvents;\n      case \"LongPolling\":\n        return HttpTransportType.LongPolling;\n      default:\n        return HttpTransportType.None;\n    }\n  }\n\n  private getLogLevel(): LogLevel {\n    switch (this.chatConfig$.value?.connectionSettings.signalRLogLevel) {\n      case \"Critical\":\n        return LogLevel.Critical; // Log level for diagnostic messages that indicate a failure that will terminate the entire application.\n      case \"Debug\":\n        return LogLevel.Debug; // Log level for low severity diagnostic messages.\n      case \"Error\":\n        return LogLevel.Error; // Log level for diagnostic messages that indicate a failure in the current operation.\n      case \"Information\":\n        return LogLevel.Information; // Log level for informational diagnostic messages.\n      case \"None\":\n        return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.\n      case \"Trace\":\n        return LogLevel.Trace; // Log level for very low severity diagnostic messages.\n      case \"Warning\":\n        return LogLevel.Warning; // Log level for diagnostic messages that indicate a non-fatal problem.\n      default:\n        return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.\n    }\n  }\n\n  get defaultOptions(): ConnectionOptions {\n    let headers: MessageHeaders = {\n      \"sinequa-force-camel-case\": \"true\",\n      \"x-language\": this.intlService.currentLocale.name,\n      \"ui-language\": this.intlService.currentLocale.name,\n    };\n    if (this.authenticationService.processedCredentials) {\n      headers = {...headers, \"sinequa-csrf-token\": this.authenticationService.processedCredentials.data.csrfToken};\n    };\n    // For the first GET request sent by signalR to start a WebSocket protocol,\n    // as far as we know, signalR only lets us tweak the request with this access token factory\n    // so we pass along the Sinequa CSRF token to pass the CSRF check..\n    return {\n      transport: this.getTransports(),\n      withCredentials: true,\n      headers,\n      accessTokenFactory: () => this.authenticationService.processedCredentials?.data?.csrfToken || \"\"\n    }\n  }\n}\n"]}
|
|
621
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-chat.service.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/websocket-chat.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from \"@angular/core\";\nimport { AuthenticationService } from \"@sinequa/core/login\";\nimport { ConnectionOptions, SignalRWebService } from \"@sinequa/core/web-services\";\nimport { Query } from \"@sinequa/core/app-utils\";\nimport { HttpTransportType, HubConnection, LogLevel, MessageHeaders } from \"@microsoft/signalr\";\nimport { ActionMessage, ActionResultEvent, ActionStartEvent, ActionStopEvent, MessageEvent, ChatMessage, ChatPayload, ChatResponse, GllmFunction, GllmModelDescription, MessageHandler, HistoryEvent, ErrorEvent, QuotaEvent, ChatContextAttachment, ContextMessageEvent, SavedChatHistory, SavedChat, ChatProgress, SuggestedActionsEvent, DebugMessageEvent, DebugMessage } from \"./types\";\nimport { ChatService } from \"./chat.service\";\nimport { merge, fromEvent, Observable, Subject, catchError, filter, forkJoin, map, shareReplay, switchMap, tap, throwError, takeUntil, take, mergeMap, of } from \"rxjs\";\n\n@Injectable()\nexport class WebSocketChatService extends ChatService {\n\n  public connection: HubConnection | undefined;\n\n  private _messageHandlers: Map<string, MessageHandler<any>> = new Map();\n  private _response: ChatMessage[];\n  private _actionMap = new Map<string, ActionMessage>();\n  private _progress: ChatProgress[] | undefined = undefined;\n  private _executionTime: string;\n  private _attachments: ChatContextAttachment[] = [];\n  private _debugMessages: DebugMessage[] = [];\n\n  public signalRService = inject(SignalRWebService);\n  public authenticationService = inject(AuthenticationService);\n\n  constructor() {\n    super();\n  }\n\n  /**\n   * Initialize the chat process after the login is complete.\n   * It includes building and starting a connection, executing parallel requests for models and functions, and handling errors during the process.\n   *\n   * @returns An Observable<boolean> indicating the success of the initialization process.\n   */\n  init(): Observable<boolean> {\n    return this.loginService.events.pipe(\n      filter((e) => e.type === 'login-complete'),\n      tap(() => this.getRequestsUrl()),\n      // Build the connection\n      switchMap(() => this.buildConnection()),\n      tap(() => this.initMessageHandlers()),\n      // Start the connection\n      switchMap(() => this.startConnection()),\n      // Execute parallel requests for models and functions\n      switchMap(() => {\n        return forkJoin([\n          this.listModels(),\n          this.listFunctions()\n        ])\n      }),\n      // Map the results of parallel requests to a boolean indicating success\n      map(([models, functions]) => {\n        const result = !!models && !!functions;\n        this.initProcess$.next(result);\n        return result;\n      }),\n      // Any errors during the process are caught, logged, and re-thrown to propagate the error further\n      catchError((error) => {\n        console.error('Error occurred:', error);\n        return throwError(() => error);\n      }),\n      // Cache and replay the emitted value for subsequent subscribers\n      shareReplay(1)\n    );\n  }\n\n  /**\n   * Define the assistant endpoint to use for the websocket requests\n   * It can be overridden by the app config\n   */\n  getRequestsUrl() {\n    if (this.chatConfig$.value!.connectionSettings.websocketEndpoint) {\n      this.REQUEST_URL = this.chatConfig$.value!.connectionSettings.websocketEndpoint;\n    } else {\n      throw new Error(`The property 'websocketEndpoint' must be provided when attempting to use 'WebSocket' in assistant instance`);\n    }\n  }\n\n  listModels(): Observable<GllmModelDescription[] | undefined> {\n    const modelsSubject$ = new Subject<GllmModelDescription[] | undefined>();\n\n    this.connection!.on('ListModels', (res) => {\n      this.models = (res.models as GllmModelDescription[] | undefined)?.filter(model => !!model.enable);\n      modelsSubject$.next(this.models);\n      modelsSubject$.complete()\n    });\n\n    // Send the request to get the list of models\n    this.connection!.invoke('ListModels', { debug: this.chatConfig$.value!.defaultValues.debug })\n      .catch(error => {\n        console.error('Error invoking ListModels:', error);\n        modelsSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return modelsSubject$.asObservable();\n  }\n\n  listFunctions(): Observable<GllmFunction[] | undefined> {\n    const functionsSubject$ = new Subject<GllmFunction[] | undefined>();\n\n    this.connection!.on('ListFunctions', (res) => {\n      this.functions = (res.functions as GllmFunction[] | undefined)?.filter(func => func.enabled);\n      functionsSubject$.next(this.functions);\n      functionsSubject$.complete()\n    });\n\n    // Send the request to get the list of functions\n    this.connection!.invoke('ListFunctions', { debug: this.chatConfig$.value!.defaultValues.debug })\n      .catch(error => {\n        console.error('Error invoking ListFunctions:', error);\n        functionsSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return functionsSubject$.asObservable();\n  }\n\n  fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse> {\n    // Start streaming by invoking the Chat method\n    this.streaming$.next(true);\n\n    // Prepare the payload to send to the Chat method\n    const data: ChatPayload = {\n      history: messages,\n      functions: this.chatConfig$.value!.defaultValues.functions?.filter(func => func.enabled).map(func => func.name),\n      debug: this.chatConfig$.value!.defaultValues.debug,\n      serviceSettings: {\n        service_id: this.chatConfig$.value!.defaultValues.service_id,\n        model_id: this.chatConfig$.value!.defaultValues.model_id,\n        top_p: this.chatConfig$.value!.defaultValues.top_p,\n        temperature: this.chatConfig$.value!.defaultValues.temperature,\n        max_tokens: this.chatConfig$.value!.defaultValues.max_tokens,\n        ...this.chatConfig$.value!.additionalServiceSettings\n      },\n      appQuery: {\n        app: this.appService.appName,\n        query\n      },\n      genericChatErrorMessage: this.chatConfig$.value!.globalSettings.genericChatErrorMessage\n    }\n    if (this.chatConfig$.value!.savedChatSettings.enabled) {\n      data.instanceId = this.chatInstanceId;\n      data.savedChatId = this.savedChatId;\n    }\n\n    // Initialize the response with an empty assistant message\n    this._response = [{role: \"assistant\", content: \"\", additionalProperties: {display: true}}] // here display: true is needed in order to be able to show the progress\n\n    // Create a Subject to signal completion\n    const completion$ = new Subject<void>();\n\n    // Create observables for each non-global handler in the _messageHandlers map (default and eventual custom ones) once it is triggered by the hub connection\n    const observables = Array\n      .from(this._messageHandlers.entries())\n      .filter(([eventName, eventHandler]) => !eventHandler.isGlobalHandler)\n      .map(([eventName, eventHandler]) => {\n        return fromEvent<any>(this.connection!, eventName).pipe(\n          mergeMap((event) => {\n            // Wrap the handler in a try-catch block to prevent the entire stream from failing if an error occurs in a single handler\n            try {\n              // Execute the handler and emit the result\n              // NB: here we could use [eventHandler.handler(event)] which behind the scenes mergeMap interprets this array as an observable sequence with one item, which it then emits\n              return of(eventHandler.handler(event));\n            } catch (error) {\n              console.error(`Error in event handler for ${eventName}:`, error);\n              // Use throwError to propagate the error downstream\n              return throwError(() => new Error(`Error in event handler for ${eventName}: ${error}`));\n            }\n          })\n        );\n      });\n\n    // Then merge them into a single observable in order to simulate the streaming behavior\n    const combined$ = merge(...observables).pipe(\n      map(() => {\n        // Define $progress from the _actionMap\n        const actions = Array.from(this._actionMap.values());\n        this._progress = actions.length > 0\n                          ? actions.map((a) => ({\n                                title: a.displayName ?? \"\",\n                                content: a.displayValue ?? \"\",\n                                done: a.executionTime !== undefined,\n                                time: a.executionTime,\n                              }))\n                          : undefined;\n\n        // Always update ONLY the first assistant message of the _response with the new $progress, $attachment and $debug\n        // Assuming that the first assistant message is always visible since the hub does not send hidden messages by design\n        // So even if the first assistant message is hidden (display: false), the _response[0] will and should contain :\n        //  - $progress, $attachment and $debug\n        //  - the content of the first visible assistant message in the workflow\n        // This is mandatory in order to match the behavior of consecutive messages and maintain consistency with the chatHistory\n        if(!!this._progress || this._attachments.length > 0 || this._debugMessages.length > 0) {\n          this._response[0].additionalProperties.$progress = this._progress;\n          this._response[0].additionalProperties.$attachment = this._attachments;\n          this._response[0].additionalProperties.$debug = this._debugMessages;\n        }\n\n        // Return the result\n        return { history: [...messages, ...this._response], executionTime: this._executionTime };\n      }),\n      takeUntil(completion$), // Complete the observable when completion$ emits\n    );\n\n    // return a new Observable that emits the result of the combined stream and handles the eventual errors of the invocation of the Chat method\n    return new Observable(observer => {\n      // Subscribe to combined stream\n      combined$.subscribe({\n        next: (value) => observer.next(value),\n        error: (err) => observer.error(err)\n      });\n\n      // Invoke the Chat method and handle errors\n      this.connection!.invoke('Chat', data)\n        .then(() => {\n          // If a valid assistant message with (display: true) was found, update it\n          // and it should always the case\n          const index = this.firstVisibleAssistantMessageIndex(this.chatHistory);\n          if (index !== -1) {\n            this.chatHistory![index].additionalProperties.$progress = this._progress;\n            this.chatHistory![index].additionalProperties.$attachment = this._attachments;\n            this.chatHistory![index].additionalProperties.$debug = this._debugMessages;\n          }\n          // Save/update the chat if savedChat enabled\n          if (this.chatConfig$.value!.savedChatSettings.enabled && this.chatHistory!.some((msg) => msg.additionalProperties?.isUserInput === true)) {\n            const action = !this.savedChatId ? this.addSavedChat(this.chatHistory!).pipe(tap(() => this.listSavedChat())) : this.updateSavedChat(this.savedChatId, undefined, this.chatHistory);\n            action.pipe(take(1)).subscribe({\n              next: () => {},\n              error: (error) => {\n                this.streaming$.next(false);\n                observer.error(error);\n              },\n              complete: () => {\n                this.streaming$.next(false);\n                observer.complete();\n              }\n            });\n          } else {\n            this.streaming$.next(false);\n            observer.complete();\n          }\n        })\n        .catch(error => {\n          console.error('Error invoking Chat:', error);\n          this.streaming$.next(false);\n          // Emit the error to the newly created observable\n          observer.error(error);\n          // Return a resolved promise to handle the error and prevent unhandled promise rejection\n          return Promise.resolve();\n        })\n        .finally(() => {\n          // This block concerns ONLY the completion of the \"Chat\" method invocation.\n          // This means the completion of the combined$ stream.\n          // It does not take into account the completion of the entire fetch method (the observable returned by fetch) and which depends on the completion of the save chat action if enabled\n          this._response = []; // Clear the _response\n          this._actionMap.clear(); // Clear the _actionMap\n          this._progress = undefined; // Clear the _progress\n          this._attachments = []; // Clear the _attachments\n          this._debugMessages = []; // Clear the _debugMessages\n          this._executionTime = \"\"; // Clear the _executionTime\n          completion$.next(); // Emit a signal to complete the observables\n          completion$.complete(); // Complete the subject\n        });\n    });\n  }\n\n  stopGeneration(): Observable<boolean> {\n    // Start stopping generation by invoking the CancelTasks method\n    this.stoppingGeneration$.next(true);\n    // Create a Subject to hold the result of the CancelTasks method\n    const stopGenerationSubject$ = new Subject<boolean>();\n\n    this.connection!.on('CancelTasks', (res) => {\n      // When the generation is stopped before streaming any VISIBLE assistant message, this means that $progress, $attachment and $debug properties will be lost.\n      // However, the \"ContextMessage\" frames will be persisted in the chatHistory and the assistant may reference them in the next generation.\n      // This leads to the problem of referencing undisplayed attachments in the next generation.\n      // To solve this problem, we need to persist $progress, $attachment and $debug properties by adding a new assistant message with empty content and these properties.\n      if (this._response.length === 1 && this._response[0].content === \"\") {\n        this.chatHistory?.push({role: \"assistant\", content: \"\", additionalProperties: {display: true, $progress: this._progress, $attachment: this._attachments, $debug: this._debugMessages}});\n      }\n      stopGenerationSubject$.next(!!res); // Emit the result of the CancelTasks method\n      stopGenerationSubject$.complete(); // Complete the subject\n      this.stoppingGeneration$.next(false); // Complete stopping generation\n    });\n\n    // Invoke the CancelTasks method and handle errors\n    this.connection!.invoke('CancelTasks')\n      .catch(error => {\n        console.error('Error invoking CancelTasks:', error);\n        stopGenerationSubject$.error(new Error(error));\n        this.stoppingGeneration$.next(false); // Complete stopping generation\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      });\n\n    return stopGenerationSubject$.asObservable();\n  }\n\n  listSavedChat(): void {\n    if (!this.chatConfig$.value!.savedChatSettings.enabled) {\n      return;\n    }\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatList', (res) => {\n      this.savedChats$.next(res.savedChats); // emits the result to the savedChats$ subject\n    });\n\n    // Invoke the method SavedChatList\n    this.connection!.invoke('SavedChatList', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatList:', error);\n        return Promise.resolve();\n      });\n  }\n\n  getSavedChat(id: string): Observable<SavedChatHistory | undefined> {\n    const savedChatSubject$ = new Subject<SavedChatHistory | undefined>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatGet', (res) => {\n      savedChatSubject$.next(res.savedChat);\n      savedChatSubject$.complete()\n    });\n\n    // Invoke the method SavedChatGet\n    this.connection!.invoke('SavedChatGet', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatGet:', error);\n        savedChatSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return savedChatSubject$.asObservable();\n  }\n\n  addSavedChat(messages: ChatMessage[]): Observable<SavedChat> {\n    const addSavedChatSubject$ = new Subject<SavedChat>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      savedChatId: this.chatId,\n      history: messages,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatAdd', (res) => {\n      this.setSavedChatId(res.savedChat.id); // Persist the savedChatId\n      this.generateAuditEvent('saved-chat.add', {}, res.savedChat.id); // Generate audit event\n      addSavedChatSubject$.next(res.savedChat);\n      addSavedChatSubject$.complete()\n    });\n\n    // Invoke the method SavedChatAdd\n    this.connection!.invoke('SavedChatAdd', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatAdd:', error);\n        addSavedChatSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return addSavedChatSubject$.asObservable();\n  }\n\n  updateSavedChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChat> {\n    const updateSavedChatSubject$ = new Subject<SavedChat>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    if(name) data[\"title\"] = name;\n    if(messages) data[\"history\"] = messages;\n\n    this.connection!.on('SavedChatUpdate', (res) => {\n      updateSavedChatSubject$.next(res.savedChat);\n      updateSavedChatSubject$.complete()\n    });\n\n    // Invoke the method SavedChatUpdate\n    this.connection!.invoke('SavedChatUpdate', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatUpdate:', error);\n        updateSavedChatSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return updateSavedChatSubject$.asObservable();\n  }\n\n  deleteSavedChat(ids: string[]): Observable<number> {\n    const deleteSavedChatSubject$ = new Subject<number>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      SavedChatIds: ids,\n      debug: this.chatConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatDelete', (res) => {\n      deleteSavedChatSubject$.next(res.deleteCount);\n      deleteSavedChatSubject$.complete();\n    });\n\n    // Invoke the method SavedChatDelete\n    this.connection!.invoke('SavedChatDelete', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatDelete:', error);\n        deleteSavedChatSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return deleteSavedChatSubject$.asObservable();\n  }\n\n  /**\n   * Initialize out-of-the-box handlers\n   * It is a placeholder for non-streaming scenarios, where you invoke a specific hub method, and the server responds with frame message(s)\n   */\n  initMessageHandlers() {\n    this.addMessageHandler(\n      \"Error\",\n      {\n        handler: (error: ErrorEvent) => {\n          console.error(error);\n          this.notificationsService.error(error);\n        },\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"Quota\",\n      {\n        handler: (message: QuotaEvent) => {\n          try {\n            this.updateQuota(message.quota)\n          } catch (error) {\n            console.error(error);\n          }\n        },\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"Debug\",\n      { handler: () => {},\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"ActionStart\",\n      { handler: (action: ActionStartEvent) => this._actionMap.set(action.guid, action),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ActionResult\",\n      {\n        handler: (action: ActionResultEvent) => this._actionMap.set(action.guid, { ...this._actionMap.get(action.guid), ...action }),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ActionStop\",\n      {\n        handler: (action: ActionStopEvent) => this._actionMap.set(action.guid, { ...this._actionMap.get(action.guid), ...action }),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ContextMessage\",\n      {\n        handler: (message: ContextMessageEvent) => this._attachments.push(message.metadata),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"Message\",\n      {\n        handler: (message: MessageEvent) => this._response.at(-1)!.content += message ?? \"\",\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"History\",\n      {\n        handler: (history: HistoryEvent) => {\n          // The ChatHistory is updated: it is the current copy concatenated with the new items ONLY (it can have multiple messages: the context messages + the response message)\n          // This is mandatory to not lose the previous updates of the chatHistory when the assistant is streaming multiple message steps\n          this.chatHistory = [...this.chatHistory!, ...(history.history.slice(this.chatHistory!.length))];\n          // Emit the updated chat usage metrics\n          if (!!this.chatHistory.at(-1)?.additionalProperties.usageMetrics) {\n            this.updateChatUsageMetrics(this.chatHistory.at(-1)!.additionalProperties.usageMetrics!);\n          }\n          this._executionTime = history.executionTime;\n        },\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"SuggestedActions\",\n      {\n        handler: (message: SuggestedActionsEvent) => {\n          // Since after the \"History\" and \"MessageBreak\" that this event is caught,\n          // $suggestedAction needs to be updated directly to the last visible \"assistant\" message in the _response and the chatHistory\n          this._response.at(-1)!.additionalProperties.$suggestedAction = (this._response.at(-1)!.additionalProperties.$suggestedAction || []).concat(message.suggestedActions);\n          const index = this.lastVisibleAssistantMessageIndex(this.chatHistory);\n          if (index !== -1) {\n            this.chatHistory![index].additionalProperties.$suggestedAction = (this.chatHistory![index].additionalProperties.$suggestedAction || []).concat(message.suggestedActions);\n          }\n        },\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"DebugDisplay\",\n      {\n        handler: (message: DebugMessageEvent) => this._debugMessages = this._debugMessages.concat(message),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"MessageBreak\",\n      {\n        handler: () => {\n          // Generate audit event\n          const details = {\n            'duration': this._executionTime,\n            'text': this.chatHistory!.at(-1)!.content,\n            'role': this.chatHistory!.at(-1)!.role, // 'assistant'\n            'rank': this.chatHistory!.length - 1,\n            'generation-tokencount': this.chatHistory!.at(-1)!.additionalProperties.usageMetrics?.completionTokenCount,\n            'prompt-tokencount': this.chatHistory!.at(-1)!.additionalProperties.usageMetrics?.promptTokenCount,\n            'attachments': JSON.stringify(this._attachments.map(({ recordId, contextId, parts, type }) => ({\n                            recordId,\n                            contextId,\n                            parts: parts.map(({ partId, text }) => ({ partId, text })),\n                            type\n                          })))\n          };\n          this.generateAuditEvent('message', details);\n          // Push a new assistant message to the _response array ONLY if the content of the last message is not empty\n          if (this._response.at(-1)!.content !== \"\") {\n            this._response.push({role: \"assistant\", content: \"\", additionalProperties: {display: true}})\n          }\n        },\n        isGlobalHandler: false\n      }\n    );\n  }\n\n  /**\n   * Override and register the entire _messageHandlers map by merging the provided map with the default one\n   * @param _messageHandlers\n   */\n  overrideMessageHandlers<T>(_messageHandlers: Map<string, MessageHandler<T>>) {\n    // Clear the already registered global chat handlers before merging the new ones\n    this._messageHandlers.forEach((eventHandler, eventName) => {\n      if(eventHandler.isGlobalHandler) {\n        this.unsubscribeMessageHandler(eventName);\n      }\n    });\n\n    // Merge the new event handlers with the existing ones\n    this._messageHandlers = new Map([...this._messageHandlers, ..._messageHandlers]);\n\n    // Register the global handlers among the merged map\n    this._messageHandlers.forEach((eventHandler, eventName) => {\n      if(eventHandler.isGlobalHandler) {\n        this.registerMessageHandler(eventName, eventHandler);\n      }\n    });\n  }\n\n  /**\n   * Add a listener for a specific event.\n   * If a listener for this same event already exists, it will be overridden.\n   * If the listener has \"isGlobalHandler\" set to true, it will be registered to the hub connection.\n   * @param eventName Name of the event to register a listener for\n   * @param eventHandler The handler to be called when the event is received\n   */\n  addMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>) {\n    this._messageHandlers.set(eventName, eventHandler);\n    if(eventHandler.isGlobalHandler) {\n      this.registerMessageHandler(eventName, eventHandler);\n    }\n  }\n\n  /**\n   * Dynamically register a listener for a specific event.\n   * If a listener for this event already exists, it will be overridden.\n   * @param eventName Name of the event to register a listener for\n   * @param eventHandler The handler to be called when the event is received\n   */\n  protected registerMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>) {\n    if (!this.connection) {\n      console.log(\"No connection found to register the listener\" + eventName);\n      return;\n    }\n\n    this.connection.on(eventName, (data: T) => {\n      eventHandler.handler(data);\n    });\n  }\n\n  /**\n   * Remove a listener for a specific event from the _messageHandlers map and unsubscribe from receiving messages for this event from the SignalR hub.\n   * @param eventName Name of the event to remove the listener for\n   */\n  removeMessageHandler(eventName: string) {\n    this._messageHandlers.delete(eventName);\n    this.unsubscribeMessageHandler(eventName);\n  }\n\n  /**\n   * Unsubscribe from receiving messages for a specific event from the SignalR hub.\n   * ALL its related listeners will be removed from hub connection\n   * This is needed to prevent accumulating old listeners when overriding the entire _messageHandlers map\n   * @param eventName Name of the event\n   */\n  protected unsubscribeMessageHandler(eventName: string) {\n    this.connection!.off(eventName);\n  }\n\n  /**\n   * Build a connection to the signalR websocket and register default listeners to the methods defined in the server hub class\n   * @param options The options for the connection. It overrides the default options\n   * @param logLevel Define the log level displayed in the console\n   * @returns Promise that resolves when the connection is built\n   */\n  buildConnection(options?: ConnectionOptions): Promise<void> {\n    return new Promise<void>((resolve, reject) => {\n      if (!this.REQUEST_URL) {\n          reject(new Error(\"No endpoint provided to connect the websocket to\"));\n          return;\n      }\n      const logLevel = this._getLogLevel();\n      this.connection = this.signalRService.buildConnection(this.REQUEST_URL, {...this.defaultOptions, ...options}, logLevel, true);\n\n      const signalRServerTimeoutInMilliseconds = this.chatConfig$.value?.connectionSettings.signalRServerTimeoutInMilliseconds;\n      if (signalRServerTimeoutInMilliseconds) {\n        this.connection.serverTimeoutInMilliseconds = signalRServerTimeoutInMilliseconds;\n      }\n\n      resolve();\n    });\n  }\n\n  /**\n   * Start the connection\n   * @returns Promise that resolves when the connection is started\n   */\n  startConnection(): Promise<void> {\n    return this.signalRService.startConnection(this.connection);\n  }\n\n  /**\n   * Stop the connection\n   * @returns Promise that resolves when the connection is stopped\n   */\n  stopConnection(): Promise<void> {\n    return this.signalRService.stopConnection(this.connection);\n  }\n\n  private _getTransports(): HttpTransportType {\n    switch (this.chatConfig$.value?.connectionSettings.signalRTransport) {\n      case \"WebSockets\":\n        return HttpTransportType.WebSockets;\n      case \"ServerSentEvents\":\n        return HttpTransportType.ServerSentEvents;\n      case \"LongPolling\":\n        return HttpTransportType.LongPolling;\n      default:\n        return HttpTransportType.None;\n    }\n  }\n\n  private _getLogLevel(): LogLevel {\n    switch (this.chatConfig$.value?.connectionSettings.signalRLogLevel) {\n      case \"Critical\":\n        return LogLevel.Critical; // Log level for diagnostic messages that indicate a failure that will terminate the entire application.\n      case \"Debug\":\n        return LogLevel.Debug; // Log level for low severity diagnostic messages.\n      case \"Error\":\n        return LogLevel.Error; // Log level for diagnostic messages that indicate a failure in the current operation.\n      case \"Information\":\n        return LogLevel.Information; // Log level for informational diagnostic messages.\n      case \"None\":\n        return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.\n      case \"Trace\":\n        return LogLevel.Trace; // Log level for very low severity diagnostic messages.\n      case \"Warning\":\n        return LogLevel.Warning; // Log level for diagnostic messages that indicate a non-fatal problem.\n      default:\n        return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.\n    }\n  }\n\n  get defaultOptions(): ConnectionOptions {\n    let headers: MessageHeaders = {\n      \"sinequa-force-camel-case\": \"true\",\n      \"x-language\": this.intlService.currentLocale.name,\n      \"ui-language\": this.intlService.currentLocale.name,\n    };\n    if (this.authenticationService.processedCredentials) {\n      headers = {...headers, \"sinequa-csrf-token\": this.authenticationService.processedCredentials.data.csrfToken};\n    };\n    // For the first GET request sent by signalR to start a WebSocket protocol,\n    // as far as we know, signalR only lets us tweak the request with this access token factory\n    // so we pass along the Sinequa CSRF token to pass the CSRF check..\n    return {\n      transport: this._getTransports(),\n      withCredentials: true,\n      headers,\n      accessTokenFactory: () => this.authenticationService.processedCredentials?.data?.csrfToken || \"\"\n    }\n  }\n}\n"]}
|