@sinequa/assistant 3.7.0 → 3.7.2
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 -8
- package/chat/chat.component.d.ts +118 -14
- package/chat/chat.service.d.ts +11 -5
- package/chat/rest-chat.service.d.ts +1 -0
- package/chat/styles/assistant.scss +2 -0
- package/chat/types.d.ts +49 -115
- package/chat/websocket-chat.service.d.ts +3 -1
- package/esm2020/chat/chat-message/chat-message.component.mjs +3 -10
- package/esm2020/chat/chat-reference/chat-reference.component.mjs +3 -3
- package/esm2020/chat/chat-settings-v3/chat-settings-v3.component.mjs +3 -3
- package/esm2020/chat/chat.component.mjs +206 -39
- package/esm2020/chat/chat.service.mjs +15 -13
- package/esm2020/chat/instance-manager.service.mjs +2 -2
- package/esm2020/chat/rest-chat.service.mjs +27 -23
- package/esm2020/chat/saved-chats/saved-chats.component.mjs +4 -4
- package/esm2020/chat/token-progress-bar/token-progress-bar.component.mjs +4 -4
- package/esm2020/chat/types.mjs +9 -10
- package/esm2020/chat/websocket-chat.service.mjs +70 -50
- package/fesm2015/sinequa-assistant-chat.mjs +328 -143
- package/fesm2015/sinequa-assistant-chat.mjs.map +1 -1
- package/fesm2020/sinequa-assistant-chat.mjs +334 -150
- package/fesm2020/sinequa-assistant-chat.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -17,6 +17,7 @@ import { TokenProgressBarComponent } from "./token-progress-bar/token-progress-b
|
|
|
17
17
|
import { DebugMessageComponent } from "./debug-message/debug-message.component";
|
|
18
18
|
import { HubConnectionState } from "@microsoft/signalr";
|
|
19
19
|
import { UtilsModule } from "@sinequa/components/utils";
|
|
20
|
+
import { NotificationsService } from "@sinequa/core/notification";
|
|
20
21
|
import * as i0 from "@angular/core";
|
|
21
22
|
import * as i1 from "@angular/common";
|
|
22
23
|
import * as i2 from "@angular/forms";
|
|
@@ -32,6 +33,7 @@ export class ChatComponent extends AbstractFacet {
|
|
|
32
33
|
this.principalService = inject(PrincipalWebService);
|
|
33
34
|
this.cdr = inject(ChangeDetectorRef);
|
|
34
35
|
this.appService = inject(AppService);
|
|
36
|
+
this.notificationsService = inject(NotificationsService);
|
|
35
37
|
/** Define the query to use to fetch answers */
|
|
36
38
|
this.query = this.searchService.query;
|
|
37
39
|
/** Define the protocol to be used for this chat instance*/
|
|
@@ -58,14 +60,12 @@ export class ChatComponent extends AbstractFacet {
|
|
|
58
60
|
this.openPreview = new EventEmitter();
|
|
59
61
|
/** Event emitter triggered when the user clicks on a suggested action */
|
|
60
62
|
this.suggestAction = new EventEmitter();
|
|
61
|
-
/** Event emitter triggered when the user clicks on a chat starter */
|
|
62
|
-
this.chatStarter = new EventEmitter();
|
|
63
63
|
this.messages$ = new BehaviorSubject(undefined);
|
|
64
64
|
this.question = '';
|
|
65
65
|
this._actions = [];
|
|
66
66
|
this._resetChatAction = new Action({
|
|
67
67
|
icon: 'fas fa-sync',
|
|
68
|
-
title: "Reset
|
|
68
|
+
title: "Reset assistant",
|
|
69
69
|
action: () => this.newChat()
|
|
70
70
|
});
|
|
71
71
|
this._sub = new Subscription();
|
|
@@ -98,7 +98,7 @@ export class ChatComponent extends AbstractFacet {
|
|
|
98
98
|
this.connection.emit(this.chatService.connection);
|
|
99
99
|
}
|
|
100
100
|
this.onLoadChat();
|
|
101
|
-
}), switchMap(_ => this.chatService.
|
|
101
|
+
}), tap(_ => this.chatService.overrideUser()), switchMap(_ => this.chatService.userOverride$), switchMap(_ => this.chatService.assistantConfig$), tap(config => {
|
|
102
102
|
this.config = config;
|
|
103
103
|
this.enabledUserInput = this.config.modeSettings.enabledUserInput;
|
|
104
104
|
this.issueTypes = this.config.auditSettings?.issueTypes?.length ? this.config.auditSettings.issueTypes : undefined;
|
|
@@ -134,13 +134,13 @@ export class ChatComponent extends AbstractFacet {
|
|
|
134
134
|
this._sub.unsubscribe();
|
|
135
135
|
this._dataSubscription?.unsubscribe();
|
|
136
136
|
this._reloadSubscription?.unsubscribe();
|
|
137
|
+
if (this.chatService instanceof WebSocketChatService) {
|
|
138
|
+
this.chatService.stopConnection();
|
|
139
|
+
}
|
|
137
140
|
}
|
|
138
141
|
get isAdmin() {
|
|
139
142
|
return this.principalService.principal?.isAdministrator || false;
|
|
140
143
|
}
|
|
141
|
-
get visibleMessagesCount() {
|
|
142
|
-
return this.messages$.value?.filter(m => m.additionalProperties.display).length || 0;
|
|
143
|
-
}
|
|
144
144
|
/**
|
|
145
145
|
* Instantiate the chat service based on the provided @input protocol
|
|
146
146
|
* This chat service instance will then be stored in the instanceManagerService with provided @input instanceId as a key
|
|
@@ -160,6 +160,17 @@ export class ChatComponent extends AbstractFacet {
|
|
|
160
160
|
this.instanceManagerService.storeInstance(this.instanceId, this.chatService);
|
|
161
161
|
}
|
|
162
162
|
get actions() { return this._actions; }
|
|
163
|
+
/**
|
|
164
|
+
* Handles the changes in the chat component.
|
|
165
|
+
* If the chat service is a WebSocketChatService, it handles the override of the message handlers if they exist.
|
|
166
|
+
* Initializes the chat with the provided chat messages if they exist, otherwise loads the default chat.
|
|
167
|
+
* If the chat is initialized, the initialization event is "Query", the query changes, and the queryChangeShouldTriggerReload function is provided,
|
|
168
|
+
* then the chat should be reloaded if the function returns true. Otherwise, the chat should be reloaded by default.
|
|
169
|
+
* It takes into account the ongoing streaming process and the ongoing stopping process to trigger that conditionally define the logic
|
|
170
|
+
* of the reload :
|
|
171
|
+
* - If the chat is streaming, then stop the generation and wait for the fetch to complete before reloading the chat.
|
|
172
|
+
* - If the chat is stopping the generation, then wait for the fetch to complete before reloading the chat.
|
|
173
|
+
*/
|
|
163
174
|
_handleChanges() {
|
|
164
175
|
const changes = this.changes$.value;
|
|
165
176
|
// If the chat service is a WebSocketChatService, handle the override of the message handlers if exists
|
|
@@ -179,11 +190,11 @@ export class ChatComponent extends AbstractFacet {
|
|
|
179
190
|
};
|
|
180
191
|
this.chatService.generateChatId();
|
|
181
192
|
if (this.chat) {
|
|
182
|
-
this.chatService.generateAuditEvent('new-chat', { 'configuration': JSON.stringify(this.chatService.
|
|
193
|
+
this.chatService.generateAuditEvent('new-chat', { 'configuration': JSON.stringify(this.chatService.assistantConfig$.value), 'chat-init': JSON.stringify(this.chat) });
|
|
183
194
|
openChat();
|
|
184
195
|
}
|
|
185
196
|
else {
|
|
186
|
-
this.chatService.generateAuditEvent('new-chat', { 'configuration': JSON.stringify(this.chatService.
|
|
197
|
+
this.chatService.generateAuditEvent('new-chat', { 'configuration': JSON.stringify(this.chatService.assistantConfig$.value) });
|
|
187
198
|
this.loadDefaultChat();
|
|
188
199
|
}
|
|
189
200
|
}
|
|
@@ -251,26 +262,52 @@ export class ChatComponent extends AbstractFacet {
|
|
|
251
262
|
}
|
|
252
263
|
}
|
|
253
264
|
}
|
|
265
|
+
/**
|
|
266
|
+
* Triggers a reload after the query change.
|
|
267
|
+
* This method performs the necessary operations to reload the chat after a query change.
|
|
268
|
+
* It sets the system and user messages, resets the savedChatId, generates a new chatId,
|
|
269
|
+
* generates a new chat audit event, and handles the query mode.
|
|
270
|
+
*/
|
|
254
271
|
_triggerReloadAfterQueryChange() {
|
|
255
272
|
const systemMsg = { role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: { display: false } };
|
|
256
273
|
const userMsg = { role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, { principal: this.principalService.principal }), additionalProperties: { display: this.config.modeSettings.displayUserPrompt } };
|
|
257
274
|
this.chatService.setSavedChatId(undefined); // Reset the savedChatId
|
|
258
275
|
this.chatService.generateChatId(); // Generate a new chatId
|
|
259
|
-
this.chatService.generateAuditEvent('new-chat', { 'configuration': JSON.stringify(this.chatService.
|
|
276
|
+
this.chatService.generateAuditEvent('new-chat', { 'configuration': JSON.stringify(this.chatService.assistantConfig$.value) }); // Generate a new chat audit event
|
|
260
277
|
this._handleQueryMode(systemMsg, userMsg);
|
|
261
278
|
}
|
|
279
|
+
/**
|
|
280
|
+
* Adds a scroll listener to the message list element.
|
|
281
|
+
* The listener is triggered when any of the following events occur:
|
|
282
|
+
* - Loading state changes
|
|
283
|
+
* - Messages change
|
|
284
|
+
* - Streaming state changes
|
|
285
|
+
* - Scroll event occurs on the message list element
|
|
286
|
+
*
|
|
287
|
+
* When the listener is triggered, it updates the `isAtBottom` property.
|
|
288
|
+
*/
|
|
262
289
|
_addScrollListener() {
|
|
263
290
|
this._sub.add(merge(this.loading$, this.messages$, this.chatService.streaming$, fromEvent(this.messageList.nativeElement, 'scroll')).subscribe(() => {
|
|
264
291
|
this.isAtBottom = this._toggleScrollButtonVisibility();
|
|
265
292
|
this.cdr.detectChanges();
|
|
266
293
|
}));
|
|
267
294
|
}
|
|
295
|
+
/**
|
|
296
|
+
* Get the model description based on the defaultValues service_id and model_id
|
|
297
|
+
*/
|
|
268
298
|
updateModelDescription() {
|
|
269
299
|
this.modelDescription = this.chatService.getModel(this.config.defaultValues.service_id, this.config.defaultValues.model_id);
|
|
270
300
|
this.cdr.detectChanges();
|
|
271
301
|
}
|
|
302
|
+
/**
|
|
303
|
+
* Submits a question from the user.
|
|
304
|
+
* If the user is editing a previous message, removes all subsequent messages from the chat history.
|
|
305
|
+
* Triggers the fetch of the answer for the submitted question by calling _fetchAnswer().
|
|
306
|
+
* Clears the input value in the UI.
|
|
307
|
+
* ⚠️ If the chat is streaming or stopping the generation, the operation is not allowed.
|
|
308
|
+
*/
|
|
272
309
|
submitQuestion() {
|
|
273
|
-
if (this.chatService.streaming$.value) {
|
|
310
|
+
if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {
|
|
274
311
|
return;
|
|
275
312
|
}
|
|
276
313
|
if (this.question.trim() && this.messages$.value && this.chatService.chatHistory) {
|
|
@@ -294,6 +331,13 @@ export class ChatComponent extends AbstractFacet {
|
|
|
294
331
|
this.questionInput.nativeElement.style.height = `auto`;
|
|
295
332
|
}
|
|
296
333
|
}
|
|
334
|
+
/**
|
|
335
|
+
* Triggers the fetch of the answer for the given question and updates the conversation.
|
|
336
|
+
* Generates an audit event for the user input.
|
|
337
|
+
*
|
|
338
|
+
* @param question - The question asked by the user.
|
|
339
|
+
* @param conversation - The current conversation messages.
|
|
340
|
+
*/
|
|
297
341
|
_fetchAnswer(question, conversation) {
|
|
298
342
|
const userMsg = { role: 'user', content: question, additionalProperties: { display: true, isUserInput: true, additionalWorkflowProperties: this.config.additionalWorkflowProperties } };
|
|
299
343
|
const messages = [...conversation, userMsg];
|
|
@@ -305,9 +349,13 @@ export class ChatComponent extends AbstractFacet {
|
|
|
305
349
|
* Depending on the connection's state :
|
|
306
350
|
* - If connected => given a list of messages, the chat endpoint is invoked for a continuation and updates the list of messages accordingly.
|
|
307
351
|
* - If any other state => a connection error message is displayed in the chat.
|
|
308
|
-
*
|
|
352
|
+
* ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.
|
|
353
|
+
* @param messages The list of messages to invoke the chat endpoint with
|
|
309
354
|
*/
|
|
310
355
|
fetch(messages) {
|
|
356
|
+
if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {
|
|
357
|
+
return;
|
|
358
|
+
}
|
|
311
359
|
this._updateConnectionStatus();
|
|
312
360
|
this.cdr.detectChanges();
|
|
313
361
|
if (this.isConnected) {
|
|
@@ -328,9 +376,7 @@ export class ChatComponent extends AbstractFacet {
|
|
|
328
376
|
// Remove the last message if it's an empty message
|
|
329
377
|
// This is due to the manner in which the chat service handles consecutive messages
|
|
330
378
|
const lastMessage = this.messages$.value?.at(-1);
|
|
331
|
-
if (lastMessage
|
|
332
|
-
&& !lastMessage?.additionalProperties?.$attachment && !lastMessage?.additionalProperties?.$progress
|
|
333
|
-
&& !lastMessage?.additionalProperties?.$debug && !lastMessage?.additionalProperties?.$suggestedAction) {
|
|
379
|
+
if (this.isEmptyAssistantMessage(lastMessage)) {
|
|
334
380
|
this.messages$.next(this.messages$.value?.slice(0, -1));
|
|
335
381
|
}
|
|
336
382
|
this.terminateFetch();
|
|
@@ -429,12 +475,18 @@ export class ChatComponent extends AbstractFacet {
|
|
|
429
475
|
this.scrollDown();
|
|
430
476
|
}
|
|
431
477
|
}
|
|
478
|
+
/**
|
|
479
|
+
* @returns true if the chat discussion is scrolled down to the bottom, false otherwise
|
|
480
|
+
*/
|
|
432
481
|
_toggleScrollButtonVisibility() {
|
|
433
482
|
if (this.messageList?.nativeElement) {
|
|
434
483
|
return Math.round(this.messageList?.nativeElement.scrollHeight - this.messageList?.nativeElement.scrollTop - 1) <= this.messageList?.nativeElement.clientHeight;
|
|
435
484
|
}
|
|
436
485
|
return true;
|
|
437
486
|
}
|
|
487
|
+
/**
|
|
488
|
+
* Scroll down to the bottom of the chat discussion
|
|
489
|
+
*/
|
|
438
490
|
scrollDown() {
|
|
439
491
|
setTimeout(() => {
|
|
440
492
|
if (this.messageList?.nativeElement) {
|
|
@@ -444,9 +496,10 @@ export class ChatComponent extends AbstractFacet {
|
|
|
444
496
|
}, 10);
|
|
445
497
|
}
|
|
446
498
|
/**
|
|
447
|
-
* Start a new chat with the defaultValues settings
|
|
448
|
-
* The savedChatId in the chat service will be reset, so that the upcoming saved chat operations will be performed on the fresh new chat
|
|
449
|
-
* If the savedChat feature is enabled, the list of saved chats will be refreshed
|
|
499
|
+
* Start a new chat with the defaultValues settings.
|
|
500
|
+
* The savedChatId in the chat service will be reset, so that the upcoming saved chat operations will be performed on the fresh new chat.
|
|
501
|
+
* If the savedChat feature is enabled, the list of saved chats will be refreshed.
|
|
502
|
+
* ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.
|
|
450
503
|
*/
|
|
451
504
|
newChat() {
|
|
452
505
|
if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {
|
|
@@ -455,9 +508,33 @@ export class ChatComponent extends AbstractFacet {
|
|
|
455
508
|
this.chatService.setSavedChatId(undefined); // Reset the savedChatId
|
|
456
509
|
this.chatService.generateChatId(); // Generate a new chatId
|
|
457
510
|
this.chatService.listSavedChat(); // Refresh the list of saved chats
|
|
458
|
-
this.chatService.generateAuditEvent('new-chat', { 'configuration': JSON.stringify(this.chatService.
|
|
511
|
+
this.chatService.generateAuditEvent('new-chat', { 'configuration': JSON.stringify(this.chatService.assistantConfig$.value) }); // Generate a new chat audit event
|
|
459
512
|
this.loadDefaultChat(); // Start a new chat
|
|
460
513
|
}
|
|
514
|
+
/**
|
|
515
|
+
* Attaches the specified document IDs to the assistant.
|
|
516
|
+
* If no document IDs are provided, the operation is not allowed.
|
|
517
|
+
* If the action for attaching a document is not defined at the application customization level, an error is logged.
|
|
518
|
+
* ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.
|
|
519
|
+
* @param ids - An array of document IDs to attach.
|
|
520
|
+
*/
|
|
521
|
+
attachToChat(ids) {
|
|
522
|
+
if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {
|
|
523
|
+
return;
|
|
524
|
+
}
|
|
525
|
+
if (!ids || ids?.length < 1) {
|
|
526
|
+
return;
|
|
527
|
+
}
|
|
528
|
+
const attachDocAction = this.config.modeSettings.actions?.["attachDocAction"];
|
|
529
|
+
if (!attachDocAction) {
|
|
530
|
+
console.error(`No action is defined for attaching a document to the assistant "${this.instanceId}"`);
|
|
531
|
+
return;
|
|
532
|
+
}
|
|
533
|
+
const userMsg = { role: 'user', content: '', additionalProperties: { display: false, isUserInput: false, type: "Action", forcedWorkflow: attachDocAction.forcedWorkflow, forcedWorkflowProperties: { ...(attachDocAction.forcedWorkflowProperties || {}), ids }, additionalWorkflowProperties: this.config.additionalWorkflowProperties } };
|
|
534
|
+
const messages = [...this.chatService.chatHistory, userMsg];
|
|
535
|
+
this.messages$.next(messages);
|
|
536
|
+
this.fetch(messages);
|
|
537
|
+
}
|
|
461
538
|
/**
|
|
462
539
|
* Start the default chat with the defaultValues settings
|
|
463
540
|
* If the chat is meant to be initialized with event === "Query", the corresponding user query message will be added to the chat history
|
|
@@ -469,15 +546,37 @@ export class ChatComponent extends AbstractFacet {
|
|
|
469
546
|
if (this.config.modeSettings.initialization.event === 'Query') {
|
|
470
547
|
this._handleQueryMode(systemMsg, userMsg);
|
|
471
548
|
}
|
|
472
|
-
else {
|
|
549
|
+
else {
|
|
550
|
+
this._handlePromptMode(systemMsg, userMsg);
|
|
551
|
+
}
|
|
552
|
+
}
|
|
553
|
+
/**
|
|
554
|
+
* Handles the prompt mode of the chat component.
|
|
555
|
+
* If `sendUserPrompt` is true, it opens the chat with both system and user messages,
|
|
556
|
+
* and generates audit events for both messages.
|
|
557
|
+
* If `sendUserPrompt` is false, it opens the chat with only the system message,
|
|
558
|
+
* and generates an audit event for the system message.
|
|
559
|
+
*
|
|
560
|
+
* @param systemMsg - The system message to be displayed in the chat.
|
|
561
|
+
* @param userMsg - The user message to be displayed in the chat (optional).
|
|
562
|
+
*/
|
|
563
|
+
_handlePromptMode(systemMsg, userMsg) {
|
|
564
|
+
if (this.config.modeSettings.sendUserPrompt) {
|
|
473
565
|
this.openChat([systemMsg, userMsg]);
|
|
474
566
|
this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));
|
|
475
567
|
this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(userMsg, 1));
|
|
476
568
|
}
|
|
569
|
+
else {
|
|
570
|
+
this.openChat([systemMsg]);
|
|
571
|
+
this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));
|
|
572
|
+
}
|
|
477
573
|
}
|
|
478
574
|
/**
|
|
575
|
+
* Handles the query mode by displaying the system message, user message, and user query message.
|
|
479
576
|
* If the provided query text is not empty, then add the user query message to the chat history and invoke the assistant
|
|
480
577
|
* Otherwise, just start a new chat with a warning message inviting the user to perform a full text search to retrieve some results
|
|
578
|
+
* @param systemMsg - The system message to be displayed.
|
|
579
|
+
* @param userMsg - The user message to be displayed.
|
|
481
580
|
*/
|
|
482
581
|
_handleQueryMode(systemMsg, userMsg) {
|
|
483
582
|
if (!!this.query.text) {
|
|
@@ -517,7 +616,7 @@ export class ChatComponent extends AbstractFacet {
|
|
|
517
616
|
*/
|
|
518
617
|
openChat(messages, savedChatId) {
|
|
519
618
|
if (!messages || !Array.isArray(messages)) {
|
|
520
|
-
console.error('Error occurs while trying to load the
|
|
619
|
+
console.error('Error occurs while trying to load the discussion. Invalid messages received :', messages);
|
|
521
620
|
return;
|
|
522
621
|
}
|
|
523
622
|
if (savedChatId) {
|
|
@@ -550,14 +649,26 @@ export class ChatComponent extends AbstractFacet {
|
|
|
550
649
|
this.question = '';
|
|
551
650
|
this.terminateFetch();
|
|
552
651
|
}
|
|
652
|
+
/**
|
|
653
|
+
* Fetch and Load the saved chat from the saved chat index.
|
|
654
|
+
* If the saved chat is found, the chat discussion will be loaded with the provided messages and chatId
|
|
655
|
+
*/
|
|
553
656
|
onLoadChat() {
|
|
554
657
|
this.loading$.next(true);
|
|
555
658
|
this._sub.add(this.chatService.loadSavedChat$
|
|
556
659
|
.pipe(filter(savedChat => !!savedChat), switchMap(savedChat => this.chatService.getSavedChat(savedChat.id)), filter(savedChatHistory => !!savedChatHistory), tap(savedChatHistory => this.openChat(savedChatHistory.history, savedChatHistory.id))).subscribe());
|
|
557
660
|
}
|
|
661
|
+
/**
|
|
662
|
+
* Stop the generation of the current assistant's answer.
|
|
663
|
+
* The fetch subscription will be terminated.
|
|
664
|
+
*/
|
|
558
665
|
stopGeneration() {
|
|
559
666
|
this.chatService.stopGeneration().subscribe(() => this.terminateFetch());
|
|
560
667
|
}
|
|
668
|
+
/**
|
|
669
|
+
* Terminate the fetch process by unsubscribing from the data subscription and updating the loading status to false.
|
|
670
|
+
* Additionally, focus on the chat input if the focusAfterResponse flag is set to true.
|
|
671
|
+
*/
|
|
561
672
|
terminateFetch() {
|
|
562
673
|
this._dataSubscription?.unsubscribe();
|
|
563
674
|
this._dataSubscription = undefined;
|
|
@@ -574,9 +685,13 @@ export class ChatComponent extends AbstractFacet {
|
|
|
574
685
|
* Thus, the user can edit and resubmit the message.
|
|
575
686
|
* Once the edited message is submitted, all subsequent messages starting from @param index will be removed from the history and the UI will be updated accordingly.
|
|
576
687
|
* The assistant will regenerate a new answer based on the updated chat history.
|
|
688
|
+
* ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.
|
|
577
689
|
* @param index The index of the user's message to edit
|
|
578
690
|
*/
|
|
579
691
|
editMessage(index) {
|
|
692
|
+
if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {
|
|
693
|
+
return;
|
|
694
|
+
}
|
|
580
695
|
this.messageToEdit = index;
|
|
581
696
|
this.remappedMessageToEdit = this._remapIndexInChatHistory(index);
|
|
582
697
|
this.question = this.chatService.chatHistory[this._remapIndexInChatHistory(index)].content;
|
|
@@ -594,9 +709,13 @@ export class ChatComponent extends AbstractFacet {
|
|
|
594
709
|
/**
|
|
595
710
|
* Starting from the provided index, remove all subsequent messages from the chat history and the UI accordingly.
|
|
596
711
|
* The assistant will regenerate a new answer based on the updated chat history.
|
|
712
|
+
* ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.
|
|
597
713
|
* @param index The index of the assistant's message to regenerate
|
|
598
714
|
*/
|
|
599
715
|
regenerateMessage(index) {
|
|
716
|
+
if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {
|
|
717
|
+
return;
|
|
718
|
+
}
|
|
600
719
|
// Update the messages in the UI by removing all subsequent 'assistant' messages starting from the provided index until the first previous 'user' message
|
|
601
720
|
let i = index;
|
|
602
721
|
while (i >= 0 && (this.messages$.value)[i].role !== 'user') {
|
|
@@ -622,14 +741,31 @@ export class ChatComponent extends AbstractFacet {
|
|
|
622
741
|
* @param index - The index to be remapped.
|
|
623
742
|
*/
|
|
624
743
|
_remapIndexInChatHistory(index) {
|
|
625
|
-
// a copy of the chat history is created to avoid modifying the original chat history.
|
|
626
|
-
|
|
627
|
-
|
|
744
|
+
// a copy of the chat history is created to avoid modifying the original chat history.
|
|
745
|
+
// Additionally, a rank is giving to each message.
|
|
746
|
+
// All messages having role 'user' are updated with the display property set to true, this is mandatory to get the correct rank of the message in the chat history when the user message to remap is hidden
|
|
747
|
+
const history = this.chatService.chatHistory
|
|
748
|
+
.slice()
|
|
749
|
+
.map((message, idx) => ({ ...message, additionalProperties: { ...message.additionalProperties, rank: idx } }))
|
|
750
|
+
.map((message) => {
|
|
751
|
+
if (message.role === "user") {
|
|
752
|
+
return { ...message, additionalProperties: { ...message.additionalProperties, display: true } };
|
|
753
|
+
}
|
|
754
|
+
return message;
|
|
628
755
|
});
|
|
629
|
-
// Count the number of hidden messages in messages$ before the provided index
|
|
756
|
+
// Count the number of hidden messages (of role different then "user") in messages$ before the provided index
|
|
757
|
+
// All messages having role 'user' are updated with the display property set to true, this is mandatory to get the correct rank of the message in the chat history when the user message to remap is hidden
|
|
630
758
|
// This is mandatory to get the correct rank of the message in the chat history
|
|
631
759
|
// Since some hidden messages (like 'system' messages) are not displayed in the UI but have been counted in the provided index
|
|
632
|
-
const numberOfHiddenMessagesInMessages$BeforeIndex = this.messages$.value
|
|
760
|
+
const numberOfHiddenMessagesInMessages$BeforeIndex = this.messages$.value
|
|
761
|
+
.slice(0, index)
|
|
762
|
+
.map((message) => {
|
|
763
|
+
if (message.role === "user") {
|
|
764
|
+
return { ...message, additionalProperties: { ...message.additionalProperties, display: true } };
|
|
765
|
+
}
|
|
766
|
+
return message;
|
|
767
|
+
})
|
|
768
|
+
.filter(message => !message.additionalProperties.display).length;
|
|
633
769
|
// remove all messages that have display set to false
|
|
634
770
|
// this is mandatory since at the point of time when the assistant answers a question,
|
|
635
771
|
// it might have some hidden messages (for example contextMessages) that are available in the chat history but don't figure in messages$ unless a new question is asked
|
|
@@ -637,6 +773,10 @@ export class ChatComponent extends AbstractFacet {
|
|
|
637
773
|
// return the index of the message in the filtered history
|
|
638
774
|
return filteredHistory[index - numberOfHiddenMessagesInMessages$BeforeIndex].additionalProperties.rank;
|
|
639
775
|
}
|
|
776
|
+
/**
|
|
777
|
+
* Handles the key up event for 'Backspace' and 'Enter' keys.
|
|
778
|
+
* @param event - The keyboard event.
|
|
779
|
+
*/
|
|
640
780
|
onKeyUp(event) {
|
|
641
781
|
switch (event.key) {
|
|
642
782
|
case 'Backspace':
|
|
@@ -653,6 +793,11 @@ export class ChatComponent extends AbstractFacet {
|
|
|
653
793
|
break;
|
|
654
794
|
}
|
|
655
795
|
}
|
|
796
|
+
/**
|
|
797
|
+
* Calculates and adjusts the height of the question input element based on its content.
|
|
798
|
+
* If the Enter key is pressed without the Shift key, it prevents the default behavior.
|
|
799
|
+
* @param event The keyboard event
|
|
800
|
+
*/
|
|
656
801
|
calculateHeight(event) {
|
|
657
802
|
if (event?.key === 'Enter' && !event.shiftKey) {
|
|
658
803
|
event?.preventDefault();
|
|
@@ -678,6 +823,8 @@ export class ChatComponent extends AbstractFacet {
|
|
|
678
823
|
this.reportComment = undefined;
|
|
679
824
|
this.reportRank = rank;
|
|
680
825
|
this.showReport = true;
|
|
826
|
+
this.chatService.chatHistory[this._remapIndexInChatHistory(rank)].additionalProperties.$liked = true;
|
|
827
|
+
this._updateChatHistory();
|
|
681
828
|
}
|
|
682
829
|
/**
|
|
683
830
|
* Send a "dislike" event on clicking on the thumb-down icon of an assistant's message.
|
|
@@ -695,6 +842,14 @@ export class ChatComponent extends AbstractFacet {
|
|
|
695
842
|
this.reportComment = undefined;
|
|
696
843
|
this.reportRank = rank;
|
|
697
844
|
this.showReport = true;
|
|
845
|
+
this.chatService.chatHistory[this._remapIndexInChatHistory(rank)].additionalProperties.$disliked = true;
|
|
846
|
+
this._updateChatHistory();
|
|
847
|
+
}
|
|
848
|
+
_updateChatHistory() {
|
|
849
|
+
this.messages$.next(this.chatService.chatHistory);
|
|
850
|
+
if (this.chatService.savedChatId) {
|
|
851
|
+
this.chatService.updateSavedChat(this.chatService.savedChatId, undefined, this.chatService.chatHistory).subscribe();
|
|
852
|
+
}
|
|
698
853
|
}
|
|
699
854
|
/**
|
|
700
855
|
* Report an issue related to the assistant's message.
|
|
@@ -712,6 +867,7 @@ export class ChatComponent extends AbstractFacet {
|
|
|
712
867
|
else {
|
|
713
868
|
this.chatService.generateAuditEvent('positive-report.send', details);
|
|
714
869
|
}
|
|
870
|
+
this.notificationsService.success('Your report has been successfully sent');
|
|
715
871
|
this.showReport = false;
|
|
716
872
|
}
|
|
717
873
|
/**
|
|
@@ -763,13 +919,6 @@ export class ChatComponent extends AbstractFacet {
|
|
|
763
919
|
this.suggestAction.emit(action);
|
|
764
920
|
this.chatService.generateAuditEvent('suggestedAction.click', { 'text': action.content, 'suggestedAction-type': action.type });
|
|
765
921
|
}
|
|
766
|
-
/**
|
|
767
|
-
* Handle the click on a chat starter.
|
|
768
|
-
* @param starter the chat starter.
|
|
769
|
-
*/
|
|
770
|
-
chatStarterClick(starter) {
|
|
771
|
-
this.chatStarter.emit(starter);
|
|
772
|
-
}
|
|
773
922
|
/**
|
|
774
923
|
* It looks for the debug messages available in the current group of "assistant" messages.
|
|
775
924
|
* By design, the debug messages are only available in the first visible message among the group "assistant" messages.
|
|
@@ -814,18 +963,38 @@ export class ChatComponent extends AbstractFacet {
|
|
|
814
963
|
}
|
|
815
964
|
return true;
|
|
816
965
|
}
|
|
966
|
+
/**
|
|
967
|
+
* Checks if the given message is an empty assistant message.
|
|
968
|
+
* An empty assistant message is defined as a message with the role 'assistant',
|
|
969
|
+
* an empty content, and no additional properties such as attachments, progress,
|
|
970
|
+
* debug information, or suggested actions.
|
|
971
|
+
*
|
|
972
|
+
* @param message - The message to check.
|
|
973
|
+
* @returns `true` if the message is an empty assistant message, `false` otherwise.
|
|
974
|
+
*/
|
|
975
|
+
isEmptyAssistantMessage(message) {
|
|
976
|
+
if (message?.role === 'assistant'
|
|
977
|
+
&& message?.content === ""
|
|
978
|
+
&& !message?.additionalProperties?.$attachment
|
|
979
|
+
&& !message?.additionalProperties?.$progress
|
|
980
|
+
&& !message?.additionalProperties?.$debug
|
|
981
|
+
&& !message?.additionalProperties?.$suggestedAction) {
|
|
982
|
+
return true;
|
|
983
|
+
}
|
|
984
|
+
return false;
|
|
985
|
+
}
|
|
817
986
|
}
|
|
818
987
|
ChatComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
819
|
-
ChatComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatComponent, isStandalone: true, selector: "sq-chat-v3", inputs: { instanceId: "instanceId", query: "query", queryChangeShouldTriggerReload: "queryChangeShouldTriggerReload", protocol: "protocol", messageHandlers: "messageHandlers", automaticScrollToLastResponse: "automaticScrollToLastResponse", focusAfterResponse: "focusAfterResponse", chat: "chat", assistantMessageIcon: "assistantMessageIcon", userMessageIcon: "userMessageIcon", connectionErrorMessageIcon: "connectionErrorMessageIcon", searchWarningMessageIcon: "searchWarningMessageIcon" }, outputs: { connection: "connection", loading$: "loading", _config: "config", data: "data", openDocument: "openDocument", openPreview: "openPreview", suggestAction: "suggestAction"
|
|
988
|
+
ChatComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatComponent, isStandalone: true, selector: "sq-chat-v3", inputs: { instanceId: "instanceId", query: "query", queryChangeShouldTriggerReload: "queryChangeShouldTriggerReload", protocol: "protocol", messageHandlers: "messageHandlers", automaticScrollToLastResponse: "automaticScrollToLastResponse", focusAfterResponse: "focusAfterResponse", chat: "chat", assistantMessageIcon: "assistantMessageIcon", userMessageIcon: "userMessageIcon", connectionErrorMessageIcon: "connectionErrorMessageIcon", searchWarningMessageIcon: "searchWarningMessageIcon" }, outputs: { connection: "connection", loading$: "loading", _config: "config", data: "data", openDocument: "openDocument", openPreview: "openPreview", suggestAction: "suggestAction" }, providers: [
|
|
820
989
|
RestChatService,
|
|
821
990
|
WebSocketChatService
|
|
822
|
-
], queries: [{ propertyName: "loadingTpl", first: true, predicate: ["loadingTpl"], descendants: true }, { propertyName: "reportTpl", first: true, predicate: ["reportTpl"], descendants: true }, { propertyName: "tokenConsumptionTpl", first: true, predicate: ["tokenConsumptionTpl"], descendants: true }, { propertyName: "debugMessagesTpl", first: true, predicate: ["debugMessagesTpl"], descendants: true }], viewQueries: [{ propertyName: "messageList", first: true, predicate: ["messageList"], descendants: true }, { propertyName: "questionInput", first: true, predicate: ["questionInput"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ng-container *ngIf=\"!initializationError\">\n <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n <!-- Token consumption -->\n <div class=\"ms-1\" *ngIf=\"config?.globalSettings?.displayUserQuotaConsumption || config?.globalSettings?.displayChatTokensConsumption\">\n <ng-container *ngTemplateOutlet=\"tokenConsumptionTpl || defaultTokenConsumptionTpl; context: { $implicit: instanceId }\"></ng-container>\n </div>\n\n <!-- Chat Messages -->\n <ul class=\"list-group list-group-flush overflow-auto flex-grow-1 pe-2 pb-2\" #messageList>\n <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n <!-- Regular messages -->\n <li class=\"list-group-item\"\n *ngIf=\"message.additionalProperties.display\"\n [style.--bs-list-group-item-padding-y.rem]=\"'0.6'\"\n [class.opacity-50]=\"messageToEdit && messageToEdit < index + 1\">\n <sq-chat-message\n [class.sq-user-message]=\"message.role === 'user'\"\n [class.last-message]=\"last\"\n [message]=\"message\"\n [conversation]=\"messages\"\n [suggestedActions]=\"last ? message.additionalProperties.$suggestedAction : undefined\"\n [chatStarters]=\"visibleMessagesCount === 1 ? config.modeSettings.initialization.chatStarters : undefined\"\n [assistantMessageIcon]=\"assistantMessageIcon\"\n [userMessageIcon]=\"userMessageIcon\"\n [connectionErrorMessageIcon]=\"connectionErrorMessageIcon\"\n [searchWarningMessageIcon]=\"searchWarningMessageIcon\"\n [streaming]=\"(chatService.streaming$ | async) && (last || isAssistantLastMessages(messages, index))\"\n [canEdit]=\"(loading$ | async) === false && ((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role === 'user'\"\n [canCopy]=\"((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role !== 'connection-error' && message.role !== 'search-warning'\"\n [canLike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n [canDislike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n [canDebug]=\"(((chatService.streaming$ | async) === false && last) || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && isAdmin && (getDebugMessages(index).length > 0) && config?.defaultValues.debug\"\n [canRegenerate]=\"(loading$ | async) === false && (((chatService.streaming$ | async) === false && last) || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && messageToEdit === undefined\"\n (edit)=\"editMessage(index)\"\n (copy)=\"copyMessage(index)\"\n (regenerate)=\"regenerateMessage(index)\"\n (openDocument)=\"openOriginalAttachment($event)\"\n (openPreview)=\"openAttachmentPreview($event)\"\n (suggestAction)=\"suggestActionClick($event, index)\"\n (chatStarterClicked)=\"chatStarterClick($event)\"\n (like)=\"onLike($event, index)\"\n (dislike)=\"onDislike($event, index)\"\n (debug)=\"showDebug(index)\">\n </sq-chat-message>\n </li>\n </ng-container>\n <!-- Loading spinner -->\n <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n </li>\n </ul>\n\n <!-- Reporting a feedback form -->\n <div class=\"issue-report bg-light pt-3 pb-2\" *ngIf=\"showReport\">\n <ng-container *ngTemplateOutlet=\"reportTpl || reportTplDefault; context: { $implicit: messageToReport, rank: reportRank, type: reportType }\"></ng-container>\n </div>\n\n <!-- User text input -->\n <div class=\"user-input mt-auto\" *ngIf=\"!showReport\">\n <div class=\"py-2\">\n <div [hidden]=\"!isConnected\">\n <ng-container *ngIf=\"enabledUserInput\" [ngTemplateOutlet]=\"inputTpl\"></ng-container>\n </div>\n <!-- Retry button -->\n <button [hidden]=\"isConnected\" class=\"btn mb-4 ast-error ast-btn sq-retry\" (click)=\"retryFetch()\">\n <span>Try again</span>\n <span *ngIf=\"retrialAttempts\" class=\"ms-2 attempts\">{{ retrialAttempts }}</span>\n </button>\n <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n {{ config?.globalSettings?.disclaimer }}\n </div>\n </div>\n </div>\n\n <!-- Floating scroll button -->\n <div *ngIf=\"!isAtBottom && !showReport\" class=\"sq-floating-scroll\" [ngClass]=\"enabledUserInput ? 'sq-floating-scroll--when-user-input' : 'sq-floating-scroll--without-user-input'\">\n <button class=\"btn shadow\" (click)=\"scrollDown()\">\n <i class=\"fas fa-angle-double-down\"></i>\n </button>\n </div>\n </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n <span class=\"visually-hidden\">Loading...</span>\n </div>\n</ng-template>\n\n<ng-template #inputTpl>\n <div class=\"px-3 py-1\">\n <div class=\"ast-input-container\">\n <button disabled class=\"btn btn-light\">\n <i class=\"fas fa-search\"></i>\n </button>\n <textarea #questionInput rows=\"1\"\n type=\"text\" class=\"form-control\"\n placeholder=\"Ask something\" autofocus\n [(ngModel)]=\"question\"\n (keyup)=\"onKeyUp($event)\"\n (keydown)=\"calculateHeight($event)\"\n [disabled]=\"(loading$ | async) || (chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\">\n </textarea>\n <button\n *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async) && !(chatService.stoppingGeneration$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Send message\"\n (click)=\"submitQuestion()\">\n <i class=\"fas fa-paper-plane\"></i>\n </button>\n <button\n *ngIf=\"messageToEdit\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Cancel edition\"\n (click)=\"messageToEdit = undefined; question = ''\">\n <i class=\"fas fa-undo-alt\"></i>\n </button>\n <span *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n Generating <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n </span>\n <span *ngIf=\"(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n Stopping <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n </span>\n <button\n *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Stop generating\"\n (click)=\"stopGeneration()\">\n <i class=\"fas fa-stop\"></i>\n </button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #reportTplDefault let-message let-rank=\"rank\" let-type=\"type\">\n <div class=\"px-3\">\n <ng-container *ngIf=\"type === 'dislike'\">\n <h5>Issue type</h5>\n <select class=\"form-select mb-4\" [(ngModel)]=\"issueType\">\n <option [value]=\"''\">Choose an issue type</option>\n <option *ngFor=\"let type of (issueTypes ?? defaultIssueTypes)\">{{type}}</option>\n </select>\n <h5>What was unsatisfying about this response? (optional)</h5>\n </ng-container>\n <ng-container *ngIf=\"type === 'like'\">\n <h5>Why did you like this answer? (optional)</h5>\n </ng-container>\n <textarea class=\"form-control\" [(ngModel)]=\"reportComment\" placeholder=\"Write your comment\"></textarea>\n <div class=\"d-flex flex-row-reverse mt-2\">\n <button class=\"btn btn-primary\" [disabled]=\"type === 'dislike' && !issueType\" (click)=\"sendReport()\">Send</button>\n <button class=\"btn btn-light\" (click)=\"ignoreReport()\">Cancel</button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultTokenConsumptionTpl let-instanceId>\n <sq-token-progress-bar\n [instanceId]=\"instanceId\">\n </sq-token-progress-bar>\n</ng-template>\n\n<div class=\"debug-messages\" [class.displayed]=\"showDebugMessages\">\n <button *ngIf=\"showDebugMessages\" class=\"btn btn-light shadow back-btn\" (click)=\"showDebugMessages=false\">\n <i class=\"fas fa-chevron-right\"></i>\n </button>\n <ng-container *ngTemplateOutlet=\"debugMessagesTpl || defaultDebugMessagesTpl; context: { $implicit: debugMessages }\">\n </ng-container>\n</div>\n\n<ng-template #defaultDebugMessagesTpl let-debugMessages>\n <sq-debug-message [data]=\"debugMessages\"></sq-debug-message>\n</ng-template>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference,:host ::ng-deep .attachment .reference{position:relative;bottom:var(--ast-reference-bottom, .3em);font-weight:var(--ast-reference-font-weight, bold);padding:var(--ast-reference-padding, 0 .2em);margin:var(--ast-reference-margin, 0 .1em);border-radius:var(--ast-reference-border-radius, .2em);background-color:var(--ast-reference-background-color, lightblue);color:var(--ast-reference-color, black)}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference{font-size:var(--ast-reference-message-font-size, .7em)}:host ::ng-deep .attachment .reference{font-size:var(--ast-reference-attachment-font-size, 13px)}:host{font-size:.875rem}:host>div>.user-input>div:not(.progress),:host>div>.issue-report>div,:host>div>ul>li{width:var(--ast-chat-container-width, 100%);max-width:100%;margin-left:auto;margin-right:auto}:host>div>ul{padding-top:var(--ast-chat-padding-top, 0);padding-bottom:var(--ast-chat-padding-bottom, 0)}li.attachment>p{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden;-webkit-line-clamp:3}li.attachment.expanded>p{display:block}.progress{--bs-progress-height: 3px}.progress.disabled{--bs-progress-height: 20px;--bs-progress-bar-bg: var(--bs-danger)}.user-input{z-index:1}.user-input ul.list-group{max-height:30vh}.form-control:disabled{background-color:#ededed}a.disabled{cursor:default;opacity:.5}.no-max-height{max-height:initial!important}.sq-floating-scroll{position:absolute;right:50%;text-align:center}.sq-floating-scroll--when-user-input{bottom:75px}.sq-floating-scroll--without-user-input{bottom:15px}.sq-floating-scroll .btn{background-color:#fff}.sq-floating-scroll .btn:hover{background-color:#fff;opacity:.9}.ast-input-container{display:flex;align-items:center;background-color:var(--ast-input-bg, #F8F8F8);border-radius:var(--ast-size-3, .75rem)}.ast-input-container>i{padding-left:var(--ast-size-3, .75rem);color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container textarea{padding-left:var(--ast-size-3, .75rem);padding-right:var(--ast-size-3, .75rem);resize:none}.ast-input-container textarea,.ast-input-container button,.ast-input-container button:hover{background-color:transparent;border:0}.ast-input-container button:hover{color:var(--ast-primary-color, #005DA7)}.ast-input-container button:not(:hover){color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container .processing{display:flex;align-items:center;color:var(--ast-secondary-color, #FF732E)}sq-chat-message.sq-user-message{float:var(--ast-user-message-float, none)}sq-token-progress-bar{z-index:10;position:absolute;top:0;right:0}.debug-messages{position:fixed;z-index:999999;right:-60%;top:0;width:60%;height:100%;transition:all .5s ease;background-color:var(--bs-body-bg);overflow:auto}.debug-messages .back-btn{position:fixed;right:0%;transition:all .5s ease}.debug-messages.displayed{right:0}.debug-messages.displayed .back-btn{right:60%}.debug-messages sq-debug-message:first-of-type{display:block;width:100%}.sq-retry{display:flex;margin:auto;font-weight:var(--font-weight-bold, 500)}.sq-retry .attempts{display:flex;border-radius:100%;background:white;height:20px;width:20px;place-content:center;align-items:center}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: ChatMessageComponent, selector: "sq-chat-message", inputs: ["message", "conversation", "suggestedActions", "chatStarters", "assistantMessageIcon", "userMessageIcon", "connectionErrorMessageIcon", "searchWarningMessageIcon", "streaming", "canEdit", "canRegenerate", "canCopy", "canDebug", "canLike", "canDislike"], outputs: ["openDocument", "openPreview", "suggestAction", "chatStarterClicked", "edit", "copy", "regenerate", "like", "dislike", "debug"] }, { kind: "component", type: TokenProgressBarComponent, selector: "sq-token-progress-bar", inputs: ["instanceId"] }, { kind: "component", type: DebugMessageComponent, selector: "sq-debug-message", inputs: ["data", "level", "parentColor"] }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i3.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
991
|
+
], queries: [{ propertyName: "loadingTpl", first: true, predicate: ["loadingTpl"], descendants: true }, { propertyName: "reportTpl", first: true, predicate: ["reportTpl"], descendants: true }, { propertyName: "tokenConsumptionTpl", first: true, predicate: ["tokenConsumptionTpl"], descendants: true }, { propertyName: "debugMessagesTpl", first: true, predicate: ["debugMessagesTpl"], descendants: true }], viewQueries: [{ propertyName: "messageList", first: true, predicate: ["messageList"], descendants: true }, { propertyName: "questionInput", first: true, predicate: ["questionInput"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<ng-container *ngIf=\"!initializationError\">\n <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n <!-- Token consumption -->\n <div class=\"ms-1\" *ngIf=\"config?.globalSettings?.displayUserQuotaConsumption || config?.globalSettings?.displayChatTokensConsumption\">\n <ng-container *ngTemplateOutlet=\"tokenConsumptionTpl || defaultTokenConsumptionTpl; context: { $implicit: instanceId }\"></ng-container>\n </div>\n\n <!-- Chat Messages -->\n <ul class=\"list-group list-group-flush overflow-auto flex-grow-1 pe-2 pb-2\" #messageList>\n <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n <!-- Regular messages -->\n <li class=\"list-group-item\"\n *ngIf=\"message.additionalProperties.display && !isEmptyAssistantMessage(message)\"\n [style.--bs-list-group-item-padding-y.rem]=\"'0.6'\"\n [class.opacity-50]=\"messageToEdit && (messageToEdit < (index + 1))\">\n <sq-chat-message\n [class.sq-user-message]=\"message.role === 'user'\"\n [class.last-message]=\"last\"\n [message]=\"message\"\n [conversation]=\"messages\"\n [suggestedActions]=\"last ? message.additionalProperties.$suggestedAction : undefined\"\n [assistantMessageIcon]=\"assistantMessageIcon\"\n [userMessageIcon]=\"userMessageIcon\"\n [connectionErrorMessageIcon]=\"connectionErrorMessageIcon\"\n [searchWarningMessageIcon]=\"searchWarningMessageIcon\"\n [streaming]=\"(chatService.streaming$ | async) && (last || isAssistantLastMessages(messages, index))\"\n [canEdit]=\"(chatService.streaming$ | async) === false && messageToEdit === undefined && message.role === 'user'\"\n [canCopy]=\"((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role !== 'connection-error' && message.role !== 'search-warning'\"\n [canLike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n [canDislike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n [canDebug]=\"(((chatService.streaming$ | async) === false && last) || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && isAdmin && (getDebugMessages(index).length > 0) && config?.defaultValues.debug\"\n [canRegenerate]=\"(chatService.streaming$ | async) === false && (last || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && messageToEdit === undefined\"\n (edit)=\"editMessage(index)\"\n (copy)=\"copyMessage(index)\"\n (regenerate)=\"regenerateMessage(index)\"\n (openDocument)=\"openOriginalAttachment($event)\"\n (openPreview)=\"openAttachmentPreview($event)\"\n (suggestAction)=\"suggestActionClick($event, index)\"\n (like)=\"onLike(message, index)\"\n (dislike)=\"onDislike(message, index)\"\n (debug)=\"showDebug(index)\">\n </sq-chat-message>\n </li>\n </ng-container>\n <!-- Loading spinner -->\n <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n </li>\n </ul>\n\n <!-- Reporting a feedback form -->\n <div class=\"issue-report pt-3 pb-2\" *ngIf=\"showReport\">\n <ng-container *ngTemplateOutlet=\"reportTpl || reportTplDefault; context: { $implicit: messageToReport, rank: reportRank, type: reportType }\"></ng-container>\n </div>\n\n <!-- User text input -->\n <div class=\"user-input mt-auto\" *ngIf=\"!showReport\">\n <div class=\"py-2\">\n <div [hidden]=\"!isConnected\">\n <ng-container *ngIf=\"enabledUserInput\" [ngTemplateOutlet]=\"inputTpl\"></ng-container>\n </div>\n <!-- Retry button -->\n <button [hidden]=\"isConnected\" class=\"btn mb-4 ast-error ast-btn sq-retry\" (click)=\"retryFetch()\">\n <span>Try again</span>\n <span *ngIf=\"retrialAttempts\" class=\"ms-2 attempts\">{{ retrialAttempts }}</span>\n </button>\n <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n {{ config?.globalSettings?.disclaimer }}\n </div>\n </div>\n </div>\n\n <!-- Floating scroll button -->\n <div *ngIf=\"!isAtBottom && !showReport\" class=\"sq-floating-scroll\" [ngClass]=\"enabledUserInput ? 'sq-floating-scroll--when-user-input' : 'sq-floating-scroll--without-user-input'\">\n <button class=\"btn shadow\" (click)=\"scrollDown()\">\n <i class=\"fas fa-angle-double-down\"></i>\n </button>\n </div>\n </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n <span class=\"visually-hidden\">Loading...</span>\n </div>\n</ng-template>\n\n<ng-template #inputTpl>\n <div class=\"px-3 py-1\">\n <div class=\"ast-input-container\">\n <button disabled class=\"btn btn-light\">\n <i class=\"fas fa-search\"></i>\n </button>\n <textarea #questionInput rows=\"1\"\n type=\"text\" class=\"form-control\"\n placeholder=\"Ask something\" autofocus\n [(ngModel)]=\"question\"\n (keyup)=\"onKeyUp($event)\"\n (keydown)=\"calculateHeight($event)\"\n [disabled]=\"(loading$ | async) || (chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\">\n </textarea>\n <button\n *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async) && !(chatService.stoppingGeneration$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Send message\"\n (click)=\"submitQuestion()\">\n <i class=\"fas fa-paper-plane\"></i>\n </button>\n <button\n *ngIf=\"messageToEdit\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Cancel edition\"\n (click)=\"messageToEdit = undefined; question = ''\">\n <i class=\"fas fa-undo-alt\"></i>\n </button>\n <span *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n Generating <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n </span>\n <span *ngIf=\"(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n Stopping <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n </span>\n <button\n *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Stop generating\"\n (click)=\"stopGeneration()\">\n <i class=\"fas fa-stop\"></i>\n </button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #reportTplDefault let-message let-rank=\"rank\" let-type=\"type\">\n <div class=\"px-3\">\n <ng-container *ngIf=\"type === 'dislike'\">\n <h5>Issue type</h5>\n <select class=\"form-select mb-4\" [(ngModel)]=\"issueType\">\n <option [value]=\"''\">Choose an issue type</option>\n <option *ngFor=\"let type of (issueTypes ?? defaultIssueTypes)\">{{type}}</option>\n </select>\n <h5>What was unsatisfying about this response? (optional)</h5>\n </ng-container>\n <ng-container *ngIf=\"type === 'like'\">\n <h5>Why did you like this answer? (optional)</h5>\n </ng-container>\n <textarea class=\"form-control\" [(ngModel)]=\"reportComment\" placeholder=\"Write your comment\"></textarea>\n <div class=\"d-flex flex-row-reverse gap-1 mt-2\">\n <button class=\"btn btn-primary\" [disabled]=\"type === 'dislike' && !issueType\" (click)=\"sendReport()\">Send</button>\n <button class=\"btn btn-light\" (click)=\"ignoreReport()\">Do not send report</button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultTokenConsumptionTpl let-instanceId>\n <sq-token-progress-bar\n [instanceId]=\"instanceId\">\n </sq-token-progress-bar>\n</ng-template>\n\n<div class=\"debug-messages\" [class.displayed]=\"showDebugMessages\">\n <button *ngIf=\"showDebugMessages\" class=\"btn btn-light shadow back-btn\" (click)=\"showDebugMessages=false\">\n <i class=\"fas fa-chevron-right\"></i>\n </button>\n <ng-container *ngTemplateOutlet=\"debugMessagesTpl || defaultDebugMessagesTpl; context: { $implicit: debugMessages }\">\n </ng-container>\n</div>\n\n<ng-template #defaultDebugMessagesTpl let-debugMessages>\n <sq-debug-message [data]=\"debugMessages\"></sq-debug-message>\n</ng-template>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference,:host ::ng-deep .attachment .reference{position:relative;bottom:var(--ast-reference-bottom, .3em);font-weight:var(--ast-reference-font-weight, bold);padding:var(--ast-reference-padding, 0 .2em);margin:var(--ast-reference-margin, 0 .1em);border-radius:var(--ast-reference-border-radius, .2em);background-color:var(--ast-reference-background-color, lightblue);color:var(--ast-reference-color, black)}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference{font-size:var(--ast-reference-message-font-size, .7em)}:host ::ng-deep .attachment .reference{font-size:var(--ast-reference-attachment-font-size, 13px)}:host{font-size:.875rem}:host>div>.user-input>div:not(.progress),:host>div>.issue-report>div,:host>div>ul>li{width:var(--ast-chat-container-width, 100%);max-width:100%;margin-left:auto;margin-right:auto}:host>div>ul{padding-top:var(--ast-chat-padding-top, 0);padding-bottom:var(--ast-chat-padding-bottom, 0)}li.attachment>p{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden;-webkit-line-clamp:3}li.attachment.expanded>p{display:block}.progress{--bs-progress-height: 3px}.progress.disabled{--bs-progress-height: 20px;--bs-progress-bar-bg: var(--bs-danger)}.user-input{z-index:1}.user-input ul.list-group{max-height:30vh}.form-control:disabled{background-color:#ededed}a.disabled{cursor:default;opacity:.5}.no-max-height{max-height:initial!important}.sq-floating-scroll{position:absolute;right:50%;text-align:center}.sq-floating-scroll--when-user-input{bottom:75px}.sq-floating-scroll--without-user-input{bottom:15px}.sq-floating-scroll .btn{background-color:#fff}.sq-floating-scroll .btn:hover{background-color:#fff;opacity:.9}.ast-input-container{display:flex;align-items:center;background-color:var(--ast-input-bg, #F8F8F8);border-radius:var(--ast-size-3, .75rem)}.ast-input-container>i{padding-left:var(--ast-size-3, .75rem);color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container textarea{padding-left:var(--ast-size-3, .75rem);padding-right:var(--ast-size-3, .75rem);resize:none}.ast-input-container textarea,.ast-input-container button,.ast-input-container button:hover{background-color:transparent;border:0}.ast-input-container button:hover{color:var(--ast-primary-color, #005DA7)}.ast-input-container button:not(:hover){color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container .processing{display:flex;align-items:center;color:var(--ast-secondary-color, #FF732E)}sq-chat-message.sq-user-message{float:var(--ast-user-message-float, none)}sq-token-progress-bar{z-index:10;position:absolute;top:0;right:0}.debug-messages{position:fixed;z-index:999999;right:-60%;top:0;width:60%;height:100%;transition:all .5s ease;background-color:var(--bs-body-bg);overflow:auto}.debug-messages .back-btn{position:fixed;right:0%;transition:all .5s ease}.debug-messages.displayed{right:0}.debug-messages.displayed .back-btn{right:60%}.debug-messages sq-debug-message:first-of-type{display:block;width:100%}.sq-retry{display:flex;margin:auto;font-weight:var(--font-weight-bold, 500)}.sq-retry .attempts{display:flex;border-radius:100%;background:white;height:20px;width:20px;place-content:center;align-items:center}.issue-report{background-color:var(--ast-report-bg, white)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: ChatMessageComponent, selector: "sq-chat-message", inputs: ["message", "conversation", "suggestedActions", "assistantMessageIcon", "userMessageIcon", "connectionErrorMessageIcon", "searchWarningMessageIcon", "streaming", "canEdit", "canRegenerate", "canCopy", "canDebug", "canLike", "canDislike"], outputs: ["openDocument", "openPreview", "suggestAction", "edit", "copy", "regenerate", "like", "dislike", "debug"] }, { kind: "component", type: TokenProgressBarComponent, selector: "sq-token-progress-bar", inputs: ["instanceId"] }, { kind: "component", type: DebugMessageComponent, selector: "sq-debug-message", inputs: ["data", "level", "parentColor"] }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i3.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
823
992
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatComponent, decorators: [{
|
|
824
993
|
type: Component,
|
|
825
994
|
args: [{ selector: 'sq-chat-v3', providers: [
|
|
826
995
|
RestChatService,
|
|
827
996
|
WebSocketChatService
|
|
828
|
-
], changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, FormsModule, ChatMessageComponent, TokenProgressBarComponent, DebugMessageComponent, UtilsModule], template: "<ng-container *ngIf=\"!initializationError\">\n <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n <!-- Token consumption -->\n <div class=\"ms-1\" *ngIf=\"config?.globalSettings?.displayUserQuotaConsumption || config?.globalSettings?.displayChatTokensConsumption\">\n <ng-container *ngTemplateOutlet=\"tokenConsumptionTpl || defaultTokenConsumptionTpl; context: { $implicit: instanceId }\"></ng-container>\n </div>\n\n <!-- Chat Messages -->\n <ul class=\"list-group list-group-flush overflow-auto flex-grow-1 pe-2 pb-2\" #messageList>\n <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n <!-- Regular messages -->\n <li class=\"list-group-item\"\n *ngIf=\"message.additionalProperties.display\"\n [style.--bs-list-group-item-padding-y.rem]=\"'0.6'\"\n [class.opacity-50]=\"messageToEdit && messageToEdit < index + 1\">\n <sq-chat-message\n [class.sq-user-message]=\"message.role === 'user'\"\n [class.last-message]=\"last\"\n [message]=\"message\"\n [conversation]=\"messages\"\n [suggestedActions]=\"last ? message.additionalProperties.$suggestedAction : undefined\"\n [chatStarters]=\"visibleMessagesCount === 1 ? config.modeSettings.initialization.chatStarters : undefined\"\n [assistantMessageIcon]=\"assistantMessageIcon\"\n [userMessageIcon]=\"userMessageIcon\"\n [connectionErrorMessageIcon]=\"connectionErrorMessageIcon\"\n [searchWarningMessageIcon]=\"searchWarningMessageIcon\"\n [streaming]=\"(chatService.streaming$ | async) && (last || isAssistantLastMessages(messages, index))\"\n [canEdit]=\"(loading$ | async) === false && ((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role === 'user'\"\n [canCopy]=\"((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role !== 'connection-error' && message.role !== 'search-warning'\"\n [canLike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n [canDislike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n [canDebug]=\"(((chatService.streaming$ | async) === false && last) || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && isAdmin && (getDebugMessages(index).length > 0) && config?.defaultValues.debug\"\n [canRegenerate]=\"(loading$ | async) === false && (((chatService.streaming$ | async) === false && last) || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && messageToEdit === undefined\"\n (edit)=\"editMessage(index)\"\n (copy)=\"copyMessage(index)\"\n (regenerate)=\"regenerateMessage(index)\"\n (openDocument)=\"openOriginalAttachment($event)\"\n (openPreview)=\"openAttachmentPreview($event)\"\n (suggestAction)=\"suggestActionClick($event, index)\"\n (chatStarterClicked)=\"chatStarterClick($event)\"\n (like)=\"onLike($event, index)\"\n (dislike)=\"onDislike($event, index)\"\n (debug)=\"showDebug(index)\">\n </sq-chat-message>\n </li>\n </ng-container>\n <!-- Loading spinner -->\n <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n </li>\n </ul>\n\n <!-- Reporting a feedback form -->\n <div class=\"issue-report bg-light pt-3 pb-2\" *ngIf=\"showReport\">\n <ng-container *ngTemplateOutlet=\"reportTpl || reportTplDefault; context: { $implicit: messageToReport, rank: reportRank, type: reportType }\"></ng-container>\n </div>\n\n <!-- User text input -->\n <div class=\"user-input mt-auto\" *ngIf=\"!showReport\">\n <div class=\"py-2\">\n <div [hidden]=\"!isConnected\">\n <ng-container *ngIf=\"enabledUserInput\" [ngTemplateOutlet]=\"inputTpl\"></ng-container>\n </div>\n <!-- Retry button -->\n <button [hidden]=\"isConnected\" class=\"btn mb-4 ast-error ast-btn sq-retry\" (click)=\"retryFetch()\">\n <span>Try again</span>\n <span *ngIf=\"retrialAttempts\" class=\"ms-2 attempts\">{{ retrialAttempts }}</span>\n </button>\n <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n {{ config?.globalSettings?.disclaimer }}\n </div>\n </div>\n </div>\n\n <!-- Floating scroll button -->\n <div *ngIf=\"!isAtBottom && !showReport\" class=\"sq-floating-scroll\" [ngClass]=\"enabledUserInput ? 'sq-floating-scroll--when-user-input' : 'sq-floating-scroll--without-user-input'\">\n <button class=\"btn shadow\" (click)=\"scrollDown()\">\n <i class=\"fas fa-angle-double-down\"></i>\n </button>\n </div>\n </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n <span class=\"visually-hidden\">Loading...</span>\n </div>\n</ng-template>\n\n<ng-template #inputTpl>\n <div class=\"px-3 py-1\">\n <div class=\"ast-input-container\">\n <button disabled class=\"btn btn-light\">\n <i class=\"fas fa-search\"></i>\n </button>\n <textarea #questionInput rows=\"1\"\n type=\"text\" class=\"form-control\"\n placeholder=\"Ask something\" autofocus\n [(ngModel)]=\"question\"\n (keyup)=\"onKeyUp($event)\"\n (keydown)=\"calculateHeight($event)\"\n [disabled]=\"(loading$ | async) || (chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\">\n </textarea>\n <button\n *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async) && !(chatService.stoppingGeneration$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Send message\"\n (click)=\"submitQuestion()\">\n <i class=\"fas fa-paper-plane\"></i>\n </button>\n <button\n *ngIf=\"messageToEdit\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Cancel edition\"\n (click)=\"messageToEdit = undefined; question = ''\">\n <i class=\"fas fa-undo-alt\"></i>\n </button>\n <span *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n Generating <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n </span>\n <span *ngIf=\"(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n Stopping <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n </span>\n <button\n *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Stop generating\"\n (click)=\"stopGeneration()\">\n <i class=\"fas fa-stop\"></i>\n </button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #reportTplDefault let-message let-rank=\"rank\" let-type=\"type\">\n <div class=\"px-3\">\n <ng-container *ngIf=\"type === 'dislike'\">\n <h5>Issue type</h5>\n <select class=\"form-select mb-4\" [(ngModel)]=\"issueType\">\n <option [value]=\"''\">Choose an issue type</option>\n <option *ngFor=\"let type of (issueTypes ?? defaultIssueTypes)\">{{type}}</option>\n </select>\n <h5>What was unsatisfying about this response? (optional)</h5>\n </ng-container>\n <ng-container *ngIf=\"type === 'like'\">\n <h5>Why did you like this answer? (optional)</h5>\n </ng-container>\n <textarea class=\"form-control\" [(ngModel)]=\"reportComment\" placeholder=\"Write your comment\"></textarea>\n <div class=\"d-flex flex-row-reverse mt-2\">\n <button class=\"btn btn-primary\" [disabled]=\"type === 'dislike' && !issueType\" (click)=\"sendReport()\">Send</button>\n <button class=\"btn btn-light\" (click)=\"ignoreReport()\">Cancel</button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultTokenConsumptionTpl let-instanceId>\n <sq-token-progress-bar\n [instanceId]=\"instanceId\">\n </sq-token-progress-bar>\n</ng-template>\n\n<div class=\"debug-messages\" [class.displayed]=\"showDebugMessages\">\n <button *ngIf=\"showDebugMessages\" class=\"btn btn-light shadow back-btn\" (click)=\"showDebugMessages=false\">\n <i class=\"fas fa-chevron-right\"></i>\n </button>\n <ng-container *ngTemplateOutlet=\"debugMessagesTpl || defaultDebugMessagesTpl; context: { $implicit: debugMessages }\">\n </ng-container>\n</div>\n\n<ng-template #defaultDebugMessagesTpl let-debugMessages>\n <sq-debug-message [data]=\"debugMessages\"></sq-debug-message>\n</ng-template>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference,:host ::ng-deep .attachment .reference{position:relative;bottom:var(--ast-reference-bottom, .3em);font-weight:var(--ast-reference-font-weight, bold);padding:var(--ast-reference-padding, 0 .2em);margin:var(--ast-reference-margin, 0 .1em);border-radius:var(--ast-reference-border-radius, .2em);background-color:var(--ast-reference-background-color, lightblue);color:var(--ast-reference-color, black)}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference{font-size:var(--ast-reference-message-font-size, .7em)}:host ::ng-deep .attachment .reference{font-size:var(--ast-reference-attachment-font-size, 13px)}:host{font-size:.875rem}:host>div>.user-input>div:not(.progress),:host>div>.issue-report>div,:host>div>ul>li{width:var(--ast-chat-container-width, 100%);max-width:100%;margin-left:auto;margin-right:auto}:host>div>ul{padding-top:var(--ast-chat-padding-top, 0);padding-bottom:var(--ast-chat-padding-bottom, 0)}li.attachment>p{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden;-webkit-line-clamp:3}li.attachment.expanded>p{display:block}.progress{--bs-progress-height: 3px}.progress.disabled{--bs-progress-height: 20px;--bs-progress-bar-bg: var(--bs-danger)}.user-input{z-index:1}.user-input ul.list-group{max-height:30vh}.form-control:disabled{background-color:#ededed}a.disabled{cursor:default;opacity:.5}.no-max-height{max-height:initial!important}.sq-floating-scroll{position:absolute;right:50%;text-align:center}.sq-floating-scroll--when-user-input{bottom:75px}.sq-floating-scroll--without-user-input{bottom:15px}.sq-floating-scroll .btn{background-color:#fff}.sq-floating-scroll .btn:hover{background-color:#fff;opacity:.9}.ast-input-container{display:flex;align-items:center;background-color:var(--ast-input-bg, #F8F8F8);border-radius:var(--ast-size-3, .75rem)}.ast-input-container>i{padding-left:var(--ast-size-3, .75rem);color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container textarea{padding-left:var(--ast-size-3, .75rem);padding-right:var(--ast-size-3, .75rem);resize:none}.ast-input-container textarea,.ast-input-container button,.ast-input-container button:hover{background-color:transparent;border:0}.ast-input-container button:hover{color:var(--ast-primary-color, #005DA7)}.ast-input-container button:not(:hover){color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container .processing{display:flex;align-items:center;color:var(--ast-secondary-color, #FF732E)}sq-chat-message.sq-user-message{float:var(--ast-user-message-float, none)}sq-token-progress-bar{z-index:10;position:absolute;top:0;right:0}.debug-messages{position:fixed;z-index:999999;right:-60%;top:0;width:60%;height:100%;transition:all .5s ease;background-color:var(--bs-body-bg);overflow:auto}.debug-messages .back-btn{position:fixed;right:0%;transition:all .5s ease}.debug-messages.displayed{right:0}.debug-messages.displayed .back-btn{right:60%}.debug-messages sq-debug-message:first-of-type{display:block;width:100%}.sq-retry{display:flex;margin:auto;font-weight:var(--font-weight-bold, 500)}.sq-retry .attempts{display:flex;border-radius:100%;background:white;height:20px;width:20px;place-content:center;align-items:center}\n"] }]
|
|
997
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, FormsModule, ChatMessageComponent, TokenProgressBarComponent, DebugMessageComponent, UtilsModule], template: "<ng-container *ngIf=\"!initializationError\">\n <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n <!-- Token consumption -->\n <div class=\"ms-1\" *ngIf=\"config?.globalSettings?.displayUserQuotaConsumption || config?.globalSettings?.displayChatTokensConsumption\">\n <ng-container *ngTemplateOutlet=\"tokenConsumptionTpl || defaultTokenConsumptionTpl; context: { $implicit: instanceId }\"></ng-container>\n </div>\n\n <!-- Chat Messages -->\n <ul class=\"list-group list-group-flush overflow-auto flex-grow-1 pe-2 pb-2\" #messageList>\n <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n <!-- Regular messages -->\n <li class=\"list-group-item\"\n *ngIf=\"message.additionalProperties.display && !isEmptyAssistantMessage(message)\"\n [style.--bs-list-group-item-padding-y.rem]=\"'0.6'\"\n [class.opacity-50]=\"messageToEdit && (messageToEdit < (index + 1))\">\n <sq-chat-message\n [class.sq-user-message]=\"message.role === 'user'\"\n [class.last-message]=\"last\"\n [message]=\"message\"\n [conversation]=\"messages\"\n [suggestedActions]=\"last ? message.additionalProperties.$suggestedAction : undefined\"\n [assistantMessageIcon]=\"assistantMessageIcon\"\n [userMessageIcon]=\"userMessageIcon\"\n [connectionErrorMessageIcon]=\"connectionErrorMessageIcon\"\n [searchWarningMessageIcon]=\"searchWarningMessageIcon\"\n [streaming]=\"(chatService.streaming$ | async) && (last || isAssistantLastMessages(messages, index))\"\n [canEdit]=\"(chatService.streaming$ | async) === false && messageToEdit === undefined && message.role === 'user'\"\n [canCopy]=\"((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role !== 'connection-error' && message.role !== 'search-warning'\"\n [canLike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n [canDislike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n [canDebug]=\"(((chatService.streaming$ | async) === false && last) || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && isAdmin && (getDebugMessages(index).length > 0) && config?.defaultValues.debug\"\n [canRegenerate]=\"(chatService.streaming$ | async) === false && (last || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && messageToEdit === undefined\"\n (edit)=\"editMessage(index)\"\n (copy)=\"copyMessage(index)\"\n (regenerate)=\"regenerateMessage(index)\"\n (openDocument)=\"openOriginalAttachment($event)\"\n (openPreview)=\"openAttachmentPreview($event)\"\n (suggestAction)=\"suggestActionClick($event, index)\"\n (like)=\"onLike(message, index)\"\n (dislike)=\"onDislike(message, index)\"\n (debug)=\"showDebug(index)\">\n </sq-chat-message>\n </li>\n </ng-container>\n <!-- Loading spinner -->\n <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n </li>\n </ul>\n\n <!-- Reporting a feedback form -->\n <div class=\"issue-report pt-3 pb-2\" *ngIf=\"showReport\">\n <ng-container *ngTemplateOutlet=\"reportTpl || reportTplDefault; context: { $implicit: messageToReport, rank: reportRank, type: reportType }\"></ng-container>\n </div>\n\n <!-- User text input -->\n <div class=\"user-input mt-auto\" *ngIf=\"!showReport\">\n <div class=\"py-2\">\n <div [hidden]=\"!isConnected\">\n <ng-container *ngIf=\"enabledUserInput\" [ngTemplateOutlet]=\"inputTpl\"></ng-container>\n </div>\n <!-- Retry button -->\n <button [hidden]=\"isConnected\" class=\"btn mb-4 ast-error ast-btn sq-retry\" (click)=\"retryFetch()\">\n <span>Try again</span>\n <span *ngIf=\"retrialAttempts\" class=\"ms-2 attempts\">{{ retrialAttempts }}</span>\n </button>\n <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n {{ config?.globalSettings?.disclaimer }}\n </div>\n </div>\n </div>\n\n <!-- Floating scroll button -->\n <div *ngIf=\"!isAtBottom && !showReport\" class=\"sq-floating-scroll\" [ngClass]=\"enabledUserInput ? 'sq-floating-scroll--when-user-input' : 'sq-floating-scroll--without-user-input'\">\n <button class=\"btn shadow\" (click)=\"scrollDown()\">\n <i class=\"fas fa-angle-double-down\"></i>\n </button>\n </div>\n </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n <span class=\"visually-hidden\">Loading...</span>\n </div>\n</ng-template>\n\n<ng-template #inputTpl>\n <div class=\"px-3 py-1\">\n <div class=\"ast-input-container\">\n <button disabled class=\"btn btn-light\">\n <i class=\"fas fa-search\"></i>\n </button>\n <textarea #questionInput rows=\"1\"\n type=\"text\" class=\"form-control\"\n placeholder=\"Ask something\" autofocus\n [(ngModel)]=\"question\"\n (keyup)=\"onKeyUp($event)\"\n (keydown)=\"calculateHeight($event)\"\n [disabled]=\"(loading$ | async) || (chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\">\n </textarea>\n <button\n *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async) && !(chatService.stoppingGeneration$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Send message\"\n (click)=\"submitQuestion()\">\n <i class=\"fas fa-paper-plane\"></i>\n </button>\n <button\n *ngIf=\"messageToEdit\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Cancel edition\"\n (click)=\"messageToEdit = undefined; question = ''\">\n <i class=\"fas fa-undo-alt\"></i>\n </button>\n <span *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n Generating <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n </span>\n <span *ngIf=\"(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n Stopping <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n </span>\n <button\n *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\"\n type=\"button\"\n class=\"btn btn-light ms-2\"\n sqTooltip=\"Stop generating\"\n (click)=\"stopGeneration()\">\n <i class=\"fas fa-stop\"></i>\n </button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #reportTplDefault let-message let-rank=\"rank\" let-type=\"type\">\n <div class=\"px-3\">\n <ng-container *ngIf=\"type === 'dislike'\">\n <h5>Issue type</h5>\n <select class=\"form-select mb-4\" [(ngModel)]=\"issueType\">\n <option [value]=\"''\">Choose an issue type</option>\n <option *ngFor=\"let type of (issueTypes ?? defaultIssueTypes)\">{{type}}</option>\n </select>\n <h5>What was unsatisfying about this response? (optional)</h5>\n </ng-container>\n <ng-container *ngIf=\"type === 'like'\">\n <h5>Why did you like this answer? (optional)</h5>\n </ng-container>\n <textarea class=\"form-control\" [(ngModel)]=\"reportComment\" placeholder=\"Write your comment\"></textarea>\n <div class=\"d-flex flex-row-reverse gap-1 mt-2\">\n <button class=\"btn btn-primary\" [disabled]=\"type === 'dislike' && !issueType\" (click)=\"sendReport()\">Send</button>\n <button class=\"btn btn-light\" (click)=\"ignoreReport()\">Do not send report</button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #defaultTokenConsumptionTpl let-instanceId>\n <sq-token-progress-bar\n [instanceId]=\"instanceId\">\n </sq-token-progress-bar>\n</ng-template>\n\n<div class=\"debug-messages\" [class.displayed]=\"showDebugMessages\">\n <button *ngIf=\"showDebugMessages\" class=\"btn btn-light shadow back-btn\" (click)=\"showDebugMessages=false\">\n <i class=\"fas fa-chevron-right\"></i>\n </button>\n <ng-container *ngTemplateOutlet=\"debugMessagesTpl || defaultDebugMessagesTpl; context: { $implicit: debugMessages }\">\n </ng-container>\n</div>\n\n<ng-template #defaultDebugMessagesTpl let-debugMessages>\n <sq-debug-message [data]=\"debugMessages\"></sq-debug-message>\n</ng-template>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-error{background-color:var(--ast-error-bg, rgba(249, 58, 55, .2));color:var(--ast-action-buttons-color, inherit)}.ast-error:hover{color:var(--ast-error-color, rgba(249, 58, 55, .7))}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black;--ast-action-buttons-color: white;--ast-action-buttons-hover-color: #6dbee6;--ast-report-bg: #070707}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference,:host ::ng-deep .attachment .reference{position:relative;bottom:var(--ast-reference-bottom, .3em);font-weight:var(--ast-reference-font-weight, bold);padding:var(--ast-reference-padding, 0 .2em);margin:var(--ast-reference-margin, 0 .1em);border-radius:var(--ast-reference-border-radius, .2em);background-color:var(--ast-reference-background-color, lightblue);color:var(--ast-reference-color, black)}:host ::ng-deep .reference,:host ::ng-deep .message-content .reference{font-size:var(--ast-reference-message-font-size, .7em)}:host ::ng-deep .attachment .reference{font-size:var(--ast-reference-attachment-font-size, 13px)}:host{font-size:.875rem}:host>div>.user-input>div:not(.progress),:host>div>.issue-report>div,:host>div>ul>li{width:var(--ast-chat-container-width, 100%);max-width:100%;margin-left:auto;margin-right:auto}:host>div>ul{padding-top:var(--ast-chat-padding-top, 0);padding-bottom:var(--ast-chat-padding-bottom, 0)}li.attachment>p{display:-webkit-box;-webkit-box-orient:vertical;overflow:hidden;-webkit-line-clamp:3}li.attachment.expanded>p{display:block}.progress{--bs-progress-height: 3px}.progress.disabled{--bs-progress-height: 20px;--bs-progress-bar-bg: var(--bs-danger)}.user-input{z-index:1}.user-input ul.list-group{max-height:30vh}.form-control:disabled{background-color:#ededed}a.disabled{cursor:default;opacity:.5}.no-max-height{max-height:initial!important}.sq-floating-scroll{position:absolute;right:50%;text-align:center}.sq-floating-scroll--when-user-input{bottom:75px}.sq-floating-scroll--without-user-input{bottom:15px}.sq-floating-scroll .btn{background-color:#fff}.sq-floating-scroll .btn:hover{background-color:#fff;opacity:.9}.ast-input-container{display:flex;align-items:center;background-color:var(--ast-input-bg, #F8F8F8);border-radius:var(--ast-size-3, .75rem)}.ast-input-container>i{padding-left:var(--ast-size-3, .75rem);color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container textarea{padding-left:var(--ast-size-3, .75rem);padding-right:var(--ast-size-3, .75rem);resize:none}.ast-input-container textarea,.ast-input-container button,.ast-input-container button:hover{background-color:transparent;border:0}.ast-input-container button:hover{color:var(--ast-primary-color, #005DA7)}.ast-input-container button:not(:hover){color:var(--ast-muted-color, rgba(33, 37, 41, .75))}.ast-input-container .processing{display:flex;align-items:center;color:var(--ast-secondary-color, #FF732E)}sq-chat-message.sq-user-message{float:var(--ast-user-message-float, none)}sq-token-progress-bar{z-index:10;position:absolute;top:0;right:0}.debug-messages{position:fixed;z-index:999999;right:-60%;top:0;width:60%;height:100%;transition:all .5s ease;background-color:var(--bs-body-bg);overflow:auto}.debug-messages .back-btn{position:fixed;right:0%;transition:all .5s ease}.debug-messages.displayed{right:0}.debug-messages.displayed .back-btn{right:60%}.debug-messages sq-debug-message:first-of-type{display:block;width:100%}.sq-retry{display:flex;margin:auto;font-weight:var(--font-weight-bold, 500)}.sq-retry .attempts{display:flex;border-radius:100%;background:white;height:20px;width:20px;place-content:center;align-items:center}.issue-report{background-color:var(--ast-report-bg, white)}\n"] }]
|
|
829
998
|
}], ctorParameters: function () { return []; }, propDecorators: { instanceId: [{
|
|
830
999
|
type: Input
|
|
831
1000
|
}], query: [{
|
|
@@ -866,8 +1035,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
|
|
|
866
1035
|
type: Output
|
|
867
1036
|
}], suggestAction: [{
|
|
868
1037
|
type: Output
|
|
869
|
-
}], chatStarter: [{
|
|
870
|
-
type: Output
|
|
871
1038
|
}], messageList: [{
|
|
872
1039
|
type: ViewChild,
|
|
873
1040
|
args: ['messageList']
|
|
@@ -887,4 +1054,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
|
|
|
887
1054
|
type: ContentChild,
|
|
888
1055
|
args: ['debugMessagesTpl']
|
|
889
1056
|
}] } });
|
|
890
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat.component.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/chat.component.ts","../../../../projects/assistant/chat/chat.component.html"],"names":[],"mappings":"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","sourcesContent":["import { inject, ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, OnInit, Output, SimpleChanges, TemplateRef, ViewChild } from \"@angular/core\";\nimport { Action } from \"@sinequa/components/action\";\nimport { AbstractFacet } from \"@sinequa/components/facet\";\nimport { SearchService } from \"@sinequa/components/search\";\nimport { AppService, Query } from \"@sinequa/core/app-utils\";\nimport { PrincipalWebService, Record as Article } from \"@sinequa/core/web-services\";\nimport { BehaviorSubject, Subscription, combineLatest, filter, fromEvent, map, merge, switchMap, take, tap } from \"rxjs\";\nimport { ChatService } from \"./chat.service\";\nimport { ChatContextAttachment, ChatConfig, ChatMessage, GllmModelDescription, MessageHandler, RawMessage, SuggestedAction, InitChat, DebugMessage, ChatStarter } from \"./types\";\nimport { InstanceManagerService } from \"./instance-manager.service\";\nimport { WebSocketChatService } from \"./websocket-chat.service\";\nimport { ChatMessageComponent } from \"./chat-message/chat-message.component\";\nimport { CommonModule } from \"@angular/common\";\nimport { FormsModule } from \"@angular/forms\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { RestChatService } from \"./rest-chat.service\";\nimport { TokenProgressBarComponent } from \"./token-progress-bar/token-progress-bar.component\";\nimport { DebugMessageComponent } from \"./debug-message/debug-message.component\";\nimport { HubConnection, HubConnectionState  } from \"@microsoft/signalr\";\nimport { UtilsModule } from \"@sinequa/components/utils\";\n\n@Component({\n  selector: 'sq-chat-v3', // mandatory since @sinequa/components already has the same tag-name \"sq-chat\"\n  templateUrl: './chat.component.html',\n  styleUrls: ['./chat.component.scss'],\n  providers: [\n    RestChatService,\n    WebSocketChatService\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [CommonModule, FormsModule, ChatMessageComponent, TokenProgressBarComponent, DebugMessageComponent, UtilsModule]\n})\nexport class ChatComponent extends AbstractFacet implements OnInit, OnChanges, OnDestroy {\n\n  public loginService = inject(LoginService);\n  public websocketService = inject(WebSocketChatService);\n  public restService = inject(RestChatService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public searchService = inject(SearchService);\n  public principalService = inject(PrincipalWebService);\n  public cdr = inject(ChangeDetectorRef);\n  public appService = inject(AppService);\n\n  /** Define the key based on it, the chat service instance will be stored */\n  @Input() instanceId: string;\n  /** Define the query to use to fetch answers */\n  @Input() query: Query = this.searchService.query;\n  /** Function that determines whether the chat should be reloaded after the query changes\n   * If not provided, the chat will be reloaded by default\n   * @param prevQuery The previous query\n   * @param newQuery The new query\n   * @returns true if the chat should be reloaded, false otherwise\n   */\n  @Input() queryChangeShouldTriggerReload: (prevQuery: Query, newQuery: Query) => boolean;\n  /** Define the protocol to be used for this chat instance*/\n  @Input() protocol: 'REST' | 'WEBSOCKET' = \"WEBSOCKET\";\n  /** Map of listeners overriding default registered ones*/\n  @Input() messageHandlers: Map<string, MessageHandler<any>> = new Map();\n  /** When the assistant answer a user question, automatically scroll down to the bottom of the discussion */\n  @Input() automaticScrollToLastResponse = false;\n  /** When the assistant answer a user question, automatically focus to the chat input */\n  @Input() focusAfterResponse = false;\n  /** A chat discussion that the component should get initialized with it */\n  @Input() chat?: InitChat;\n  /** Icon to use for the assistant messages */\n  @Input() assistantMessageIcon = 'sq-sinequa';\n  /** Icon to use for the user messages */\n  @Input() userMessageIcon: string;\n  /** Icon to use for the connection error messages */\n  @Input() connectionErrorMessageIcon: string;\n  /** Icon to use for the search warning messages */\n  @Input() searchWarningMessageIcon: string;\n  /** Event emitter triggered once the signalR connection is established  */\n  @Output() connection = new EventEmitter<HubConnection>();\n  /** Event emitter triggered each time the assistant updates the current chat */\n  /** Event emitter triggered when the chat is loading new content */\n  @Output(\"loading\") loading$ = new EventEmitter<boolean>(false);\n  /** Emits the assistant configuration used when instantiating the component */\n  @Output(\"config\") _config = new EventEmitter<ChatConfig>();\n  @Output() data = new EventEmitter<ChatMessage[]>();\n  /** Event emitter triggered when the user clicks to open the original document representing the context attachment*/\n  @Output() openDocument = new EventEmitter<Article>();\n  /** Event emitter triggered when the user clicks to open the preview of a document representing the context attachment */\n  @Output() openPreview = new EventEmitter<ChatContextAttachment>();\n  /** Event emitter triggered when the user clicks on a suggested action */\n  @Output() suggestAction = new EventEmitter<SuggestedAction>();\n  /** Event emitter triggered when the user clicks on a chat starter */\n  @Output() chatStarter = new EventEmitter<ChatStarter>();\n\n  @ViewChild('messageList') messageList?: ElementRef<HTMLUListElement>;\n  @ViewChild('questionInput') questionInput?: ElementRef<HTMLTextAreaElement>;\n\n  @ContentChild('loadingTpl') loadingTpl?: TemplateRef<any>;\n  @ContentChild('reportTpl') reportTpl?: TemplateRef<any>;\n  @ContentChild('tokenConsumptionTpl') tokenConsumptionTpl?: TemplateRef<any>;\n  @ContentChild('debugMessagesTpl') debugMessagesTpl?: TemplateRef<any>;\n\n  chatService: ChatService;\n  config: ChatConfig;\n  messages$ = new BehaviorSubject<ChatMessage[] | undefined>(undefined);\n\n  question = '';\n\n  _actions: Action[] = [];\n  private _resetChatAction = new Action({\n    icon: 'fas fa-sync',\n    title: \"Reset chat\",\n    action: () => this.newChat()\n  });\n\n  private _sub = new Subscription();\n  private _dataSubscription: Subscription | undefined;\n\n  /** Variables that depend on the type of model in use */\n  modelDescription?: GllmModelDescription;\n\n  messageToEdit?: number;\n  remappedMessageToEdit?: number;\n  changes$ = new BehaviorSubject<SimpleChanges | undefined>(undefined);\n  currentMessageIndex: number | undefined;\n  firstChangesHandled = false;\n  isAtBottom = true;\n  initializationError = false;\n  enabledUserInput = false;\n  isConnected = true; // By default, the chat is considered connected\n  retrialAttempts: number | undefined;\n  // Flag to track whether the 'reconnected' listener is already registered\n  private _isReconnectedListenerRegistered = false;\n\n  // Issue reporting\n  issueTypes?: string[];\n  defaultIssueTypes: string[] = [\n    'User Interface bug',\n    'Incorrect or misleading response',\n    'Incomplete response',\n    'Technical issue',\n    'Privacy/data security issue',\n    'Other'\n  ];\n  issueType: string = '';\n  reportComment?: string;\n  messageToReport?: ChatMessage;\n  reportRank?: number;\n  reportType: 'like' | 'dislike' = 'dislike';\n  showReport = false;\n\n  // Debug messages\n  debugMessages: DebugMessage[] | undefined;\n  showDebugMessages = false;\n\n  private _previousQuery: Query;\n  private _reloadSubscription: Subscription | undefined = undefined;\n\n  constructor() {\n    super();\n    this._actions.push(this._resetChatAction);\n  }\n\n  ngOnInit(): void {\n    this._sub.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        map(_ => this.chatService.initChatConfig()),\n        switchMap(() => this.chatService.initConfig$),\n        filter(initConfig => !!initConfig),\n        switchMap(_ => this.chatService.init()),\n        switchMap(_ => this.chatService.initProcess$),\n        filter(success => !!success),\n        tap(_ => {\n          if (this.chatService instanceof WebSocketChatService) {\n            this.connection.emit(this.chatService.connection);\n          }\n          this.onLoadChat();\n        }),\n        switchMap(_ => this.chatService.chatConfig$),\n        tap(config => {\n          this.config = config!;\n          this.enabledUserInput = this.config.modeSettings.enabledUserInput;\n          this.issueTypes = this.config.auditSettings?.issueTypes?.length ? this.config.auditSettings!.issueTypes : undefined;\n          this._config.emit(config);\n          try {\n            this.updateModelDescription();\n            if(!this.firstChangesHandled) {\n              this._previousQuery = JSON.parse(JSON.stringify(this.query)); // Initialize the previous query\n              this._handleChanges();\n              this._addScrollListener();\n              this.firstChangesHandled = true;\n            }\n          } catch (error) {\n            this.initializationError = true\n            throw error;\n          }\n        })\n      ).subscribe()\n    );\n\n    this._sub.add(\n      combineLatest([\n        this.chatService.streaming$,\n        this.chatService.stoppingGeneration$\n      ]).pipe(\n        map(([streaming, stoppingGeneration]) => !!(streaming || stoppingGeneration))\n      ).subscribe((result) => {\n        this._resetChatAction.disabled = result;\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    this.changes$.next(changes);\n    if (this.config) {\n      this._handleChanges();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this._sub.unsubscribe();\n    this._dataSubscription?.unsubscribe();\n    this._reloadSubscription?.unsubscribe();\n  }\n\n  get isAdmin(): boolean {\n    return this.principalService.principal?.isAdministrator || false;\n  }\n\n  get visibleMessagesCount(): number {\n    return this.messages$.value?.filter(m => m.additionalProperties.display).length || 0;\n  }\n\n  /**\n   * Instantiate the chat service based on the provided @input protocol\n   * This chat service instance will then be stored in the instanceManagerService with provided @input instanceId as a key\n   */\n  instantiateChatService(): void {\n    switch (this.protocol) {\n      case 'REST':\n        this.chatService = this.restService;\n        break;\n      case 'WEBSOCKET':\n        this.chatService = this.websocketService;\n        break;\n      default:\n        throw new Error(`Could not found a ChatService implementation corresponding to the provided protocol: '${this.protocol}'`);\n    }\n    this.chatService.setChatInstanceId(this.instanceId);\n    this.instanceManagerService.storeInstance(this.instanceId, this.chatService);\n  }\n\n  override get actions() { return this._actions; }\n\n  private _handleChanges() {\n    const changes = this.changes$.value;\n    // If the chat service is a WebSocketChatService, handle the override of the message handlers if exists\n    if (changes?.messageHandlers && this.messageHandlers && this.chatService instanceof WebSocketChatService) {\n      this.chatService.overrideMessageHandlers(this.messageHandlers);\n    }\n    /**\n     * Initialize the chat with the provided chat messages if exists, otherwise load the default chat\n     * Once the chat is initialized (firstChangesHandled is true), allow opening the chat with the new provided messages (if exists)\n     */\n    if (!this.firstChangesHandled || changes?.chat) {\n      const openChat = () => {\n        if (this.messages$.value) {\n          this.chatService.listSavedChat(); // Refresh the list of saved chats\n        }\n        this.openChat(this.chat!.messages);\n      };\n      this.chatService.generateChatId();\n      if (this.chat) {\n        this.chatService.generateAuditEvent('new-chat', {'configuration': JSON.stringify(this.chatService.chatConfig$.value),'chat-init': JSON.stringify(this.chat)});\n        openChat();\n      } else {\n        this.chatService.generateAuditEvent('new-chat', {'configuration': JSON.stringify(this.chatService.chatConfig$.value)});\n        this.loadDefaultChat();\n      }\n    }\n    /**\n     * If the chat is initialized, the initialization event is \"Query\", the query changes and the queryChangeShouldTriggerReload function is provided,\n     * then the chat should be reloaded if the function returns true\n     * Otherwise, the chat should be reloaded by default\n     */\n    if (this.firstChangesHandled && changes?.query && this.config.modeSettings.initialization.event === 'Query') {\n      if (this.queryChangeShouldTriggerReload ? this.queryChangeShouldTriggerReload(this._previousQuery, this.query) : true) {\n        if (!!this.chatService.stoppingGeneration$.value) {\n          if (!this._reloadSubscription) {\n            // Create a subscription to wait for both streaming$ and stoppingGeneration$ to be false\n            this._reloadSubscription = combineLatest([\n              this.chatService.streaming$,\n              this.chatService.stoppingGeneration$\n            ])\n            .pipe(\n              filter(([streaming, stopping]) => !streaming && !stopping), // Wait until both are false\n              take(1) // Complete after the first match\n            ).subscribe(() => {\n              // Execute the reload after the query change\n              this._triggerReloadAfterQueryChange();\n              // Update _previousQuery with the current query\n              this._previousQuery = JSON.parse(JSON.stringify(this.query));\n              // Clean up subscription and reset its value\n              this._reloadSubscription = undefined;\n            });\n          }\n        } else if (!!this.chatService.streaming$.value) {\n          if (!this._reloadSubscription) {\n            this._reloadSubscription = this.chatService.stopGeneration()\n            .subscribe({\n              next: () => {},\n              error: () => {\n                // Clean up subscription and reset its value\n                this._reloadSubscription?.unsubscribe();\n                this._reloadSubscription = undefined;\n              },\n              complete: () => {\n                // Wait for the ongoing fetch to complete, then trigger the reload\n                this.chatService.streaming$.pipe(\n                  filter((streaming) => !streaming),\n                  take(1)\n                ).subscribe(() => {\n                  // Execute the reload after the query change\n                  this._triggerReloadAfterQueryChange();\n                  // Update _previousQuery with the current query\n                  this._previousQuery = JSON.parse(JSON.stringify(this.query));\n                  // Clean up subscription and reset its value\n                  this._reloadSubscription!.unsubscribe();\n                  this._reloadSubscription = undefined;\n                });\n              }\n            });\n          }\n        } else {\n          // Execute the reload after the query change\n          this._triggerReloadAfterQueryChange();\n          // Update _previousQuery with the current query\n          this._previousQuery = JSON.parse(JSON.stringify(this.query));\n        }\n      } else {\n        // Update _previousQuery with the current query\n        this._previousQuery = JSON.parse(JSON.stringify(this.query));\n      }\n    }\n  }\n\n  private _triggerReloadAfterQueryChange() {\n    const systemMsg = {role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: {display: false}};\n    const userMsg = {role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, {principal: this.principalService.principal}), additionalProperties: {display: this.config.modeSettings.displayUserPrompt}};\n    this.chatService.setSavedChatId(undefined); // Reset the savedChatId\n    this.chatService.generateChatId(); // Generate a new chatId\n    this.chatService.generateAuditEvent('new-chat', {'configuration': JSON.stringify(this.chatService.chatConfig$.value)}); // Generate a new chat audit event\n    this._handleQueryMode(systemMsg, userMsg);\n  }\n\n  private _addScrollListener() {\n    this._sub.add(\n      merge(this.loading$, this.messages$, this.chatService.streaming$, fromEvent(this.messageList!.nativeElement, 'scroll')).subscribe(() => {\n        this.isAtBottom = this._toggleScrollButtonVisibility();\n        this.cdr.detectChanges();\n      })\n    );\n  }\n\n  updateModelDescription() {\n    this.modelDescription = this.chatService.getModel(this.config.defaultValues.service_id, this.config.defaultValues.model_id);\n    this.cdr.detectChanges();\n  }\n\n  submitQuestion() {\n    if (this.chatService.streaming$.value) {\n      return;\n    }\n    if(this.question.trim() && this.messages$.value && this.chatService.chatHistory) {\n      // When the user submits a question, if the user is editing a previous message, remove all subsequent messages from the chat history\n      if (this.messageToEdit !== undefined) {\n        // Update the messages in the UI\n        this.messages$.next(this.messages$.value.slice(0, this.messageToEdit));\n        // Update the raw messages in the chat history which is the clean version used to make the next request\n        this.chatService.chatHistory = this.chatService.chatHistory.slice(0, this.remappedMessageToEdit);\n        this.messageToEdit = undefined;\n        this.remappedMessageToEdit = undefined;\n      }\n      // Remove the search warning message if exists\n      if (this.chatService.chatHistory.at(-1)?.role === 'search-warning') {\n        this.chatService.chatHistory.pop();\n      }\n      // Fetch the answer\n      this._fetchAnswer(this.question.trim(), this.chatService.chatHistory);\n      // Clear the input value in the UI\n      this.questionInput!.nativeElement.value = '';\n      this.questionInput!.nativeElement.style.height = `auto`;\n    }\n  }\n\n  private _fetchAnswer(question: string, conversation: ChatMessage[]) {\n    const userMsg = {role: 'user', content: question, additionalProperties: {display: true, isUserInput: true, additionalWorkflowProperties: this.config.additionalWorkflowProperties}};\n    const messages = [...conversation, userMsg];\n    this.messages$.next(messages);\n    this.fetch(messages);\n    this.chatService.generateAuditEvent('message', {...this._defineMessageAuditDetails(userMsg, messages.length - 1), 'query': JSON.stringify(this.query), 'is-user-input': true, 'enabled-functions': this.config.defaultValues.functions?.filter(func => func.enabled).map(func => func.name), 'additional-workflow-properties': JSON.stringify(this.config.additionalWorkflowProperties)});\n  }\n\n  /**\n   * Depending on the connection's state :\n   *  - If connected => given a list of messages, the chat endpoint is invoked for a continuation and updates the list of messages accordingly.\n   *  - If any other state => a connection error message is displayed in the chat.\n   * @param messages\n   */\n  public fetch(messages: ChatMessage[]) {\n    this._updateConnectionStatus();\n    this.cdr.detectChanges();\n\n    if (this.isConnected) {\n      this.loading$.next(true);\n      this._dataSubscription?.unsubscribe();\n      this._dataSubscription = this.chatService.fetch(messages, this.query)\n        .subscribe({\n          next: res => this.updateData(res.history),\n          error: () => {\n            this._updateConnectionStatus();\n            if (!this.isConnected) {\n              const message = {role: 'connection-error', content: this.config.connectionSettings.connectionErrorMessage, additionalProperties: {display: true}};\n              this.messages$.next([...messages, message]);\n            }\n            this.terminateFetch();\n          },\n          complete: () => {\n            // Remove the last message if it's an empty message\n            // This is due to the manner in which the chat service handles consecutive messages\n            const lastMessage = this.messages$.value?.at(-1);\n            if (lastMessage?.role === 'assistant' && lastMessage?.content === \"\"\n                && !lastMessage?.additionalProperties?.$attachment && !lastMessage?.additionalProperties?.$progress\n                && !lastMessage?.additionalProperties?.$debug && !lastMessage?.additionalProperties?.$suggestedAction) {\n                  this.messages$.next(this.messages$.value?.slice(0, -1));\n            }\n            this.terminateFetch();\n          }\n        });\n    } else {\n      const message = {role: 'connection-error', content: this.config.connectionSettings.connectionErrorMessage, additionalProperties: {display: true}};\n      this.messages$.next([...messages, message]);\n    }\n\n    if(this.automaticScrollToLastResponse) {\n      this.scrollDown();\n    }\n  }\n\n  /**\n   * Retry to fetch the messages if the connection issues.\n   *  - If reconnecting => keep display the connection error message even when clicking on the \"retry\" button and increasing the number of retrial attempts, until the connection is re-established\n   *  - If disconnected => On click on the \"retry\" button, start the connection process while displaying the connection error message :\n   *      * If successful => given a list of messages, the chat endpoint is invoked for a continuation and updates the list of messages accordingly.\n   *      * If failed => increase the number of retrial attempts\n   */\n  retryFetch() {\n    if(this.chatService instanceof WebSocketChatService) {\n      // A one-time listener for reconnected event\n      const onReconnectedHandler = () => {\n        // Get the messages without the last one (the connection error message)\n        const messages = this.messages$.value!.slice(0, -1);\n        // Find the last \"user\" message in the messages list\n        let index = messages.length - 1;\n        while (index >= 0 && messages[index].role !== 'user') {\n          index--;\n        }\n        // If a user message is found (and it should always be the case), remove all subsequent messages from the chat history\n        // Update the messages in the UI\n        // and fetch the answer from the assistant\n        if (index >= 0) {\n          this.messages$.next(this.messages$.value!.slice(0, index+1));\n          const remappedIndex = this._remapIndexInChatHistory(index);\n          this.chatService.chatHistory = this.chatService.chatHistory!.slice(0, remappedIndex+1);\n          this.fetch(this.chatService.chatHistory);\n        }\n        this.retrialAttempts = undefined; // Reset the number of retrial attempts\n        /**\n         * To remove the handler for onreconnected() after it's been registered,cannot directly use off() like you would for normal events registered with connection.on().\n         * Instead, you need to explicitly remove or reset the handler by assigning it to null or an empty function\n         */\n        (this.chatService as WebSocketChatService).connection!.onreconnected(() => {});\n        // Reset the flag to ensure the handler is registered again when needed\n        this._isReconnectedListenerRegistered = false;\n      }\n      // Depending on the connection's state, take the appropriate action\n      switch (this.chatService.connection!.state) {\n        case HubConnectionState.Connected:\n          // If the connection is re-established in the meantime, fetch the messages\n          onReconnectedHandler();\n          break;\n        case HubConnectionState.Reconnecting:\n          // Attach the reconnected listener if not already registered\n          if (!this._isReconnectedListenerRegistered) {\n            this.chatService.connection!.onreconnected(onReconnectedHandler);\n            this._isReconnectedListenerRegistered = true;\n          }\n          // Increase the number of retrial attempts\n          this.retrialAttempts = this.retrialAttempts ? this.retrialAttempts+1 : 1;\n          break;\n        case HubConnectionState.Disconnected:\n          // Start the new connection\n          this.chatService.startConnection()\n          .then(() => onReconnectedHandler())\n          .catch(() => { // If the connection fails, increase the number of retrial attempts\n            this.retrialAttempts = this.retrialAttempts ? this.retrialAttempts+1 : 1;\n          });\n          break;\n        default:\n          break;\n      }\n    }\n  }\n\n  /**\n   * Check if the signalR connection is connected.\n   * For the REST protocol, the connection is always considered connected (for the moment).\n   */\n  private _updateConnectionStatus() {\n    this.isConnected = (this.chatService instanceof WebSocketChatService) ? this.chatService.connection!.state === HubConnectionState.Connected : true;\n  }\n\n  /**\n   * Update the UI with the new messages\n   * @param messages\n   */\n  updateData(messages: ChatMessage[]) {\n    this.messages$.next(messages);\n    this.data.emit(messages);\n    this.loading$.next(false);\n    this.question = '';\n    if(this.automaticScrollToLastResponse) {\n      this.scrollDown();\n    }\n  }\n\n  private _toggleScrollButtonVisibility(): boolean {\n    if(this.messageList?.nativeElement) {\n      return Math.round(this.messageList?.nativeElement.scrollHeight - this.messageList?.nativeElement.scrollTop - 1) <= this.messageList?.nativeElement.clientHeight;\n    }\n    return true;\n  }\n\n  scrollDown() {\n    setTimeout(() => {\n      if(this.messageList?.nativeElement) {\n        this.messageList.nativeElement.scrollTop = this.messageList.nativeElement.scrollHeight;\n        this.cdr.detectChanges();\n      }\n    }, 10);\n  }\n\n  /**\n   * Start a new chat with the defaultValues settings\n   * The savedChatId in the chat service will be reset, so that the upcoming saved chat operations will be performed on the fresh new chat\n   * If the savedChat feature is enabled, the list of saved chats will be refreshed\n   */\n  newChat() {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.chatService.setSavedChatId(undefined); // Reset the savedChatId\n    this.chatService.generateChatId(); // Generate a new chatId\n    this.chatService.listSavedChat(); // Refresh the list of saved chats\n    this.chatService.generateAuditEvent('new-chat', {'configuration': JSON.stringify(this.chatService.chatConfig$.value)}); // Generate a new chat audit event\n    this.loadDefaultChat(); // Start a new chat\n  }\n\n  /**\n   * Start the default chat with the defaultValues settings\n   * If the chat is meant to be initialized with event === \"Query\", the corresponding user query message will be added to the chat history\n   */\n  loadDefaultChat() {\n    // Define the default system prompt and user prompt messages\n    const systemMsg = {role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: {display: false}};\n    const userMsg = {role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, {principal: this.principalService.principal}), additionalProperties: {display: this.config.modeSettings.displayUserPrompt}};\n\n    if (this.config.modeSettings.initialization.event === 'Query') {\n      this._handleQueryMode(systemMsg, userMsg);\n    } else { // If the chat is meant to be initialized with event === \"Prompt\"\n      this.openChat([systemMsg, userMsg]);\n      this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));\n      this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(userMsg, 1));\n    }\n  }\n\n  /**\n   * If the provided query text is not empty, then add the user query message to the chat history and invoke the assistant\n   * Otherwise, just start a new chat with a warning message inviting the user to perform a full text search to retrieve some results\n   */\n  private _handleQueryMode(systemMsg: ChatMessage, userMsg: ChatMessage) {\n    if (!!this.query.text) {\n      const userQueryMsg = {role: 'user', content: this.query.text, additionalProperties: {display: this.config.modeSettings.initialization.displayUserQuery, query: this.query, forcedWorkflow: this.config.modeSettings.initialization.forcedWorkflow, isUserInput: true, additionalWorkflowProperties: this.config.additionalWorkflowProperties}};\n      if (this.config.modeSettings.sendUserPrompt) {\n        this.openChat([systemMsg, userMsg, userQueryMsg]);\n        this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));\n        this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(userMsg, 1));\n        this.chatService.generateAuditEvent('message', {...this._defineMessageAuditDetails(userQueryMsg, 2), 'query': JSON.stringify(this.query) ,'is-user-input': true, 'forced-workflow': this.config.modeSettings.initialization.forcedWorkflow, 'enabled-functions': this.config.defaultValues.functions?.filter(func => func.enabled).map(func => func.name), 'additional-workflow-properties':JSON.stringify(this.config.additionalWorkflowProperties)});\n      } else {\n        this.openChat([systemMsg, userQueryMsg]);\n        this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));\n        this.chatService.generateAuditEvent('message', {...this._defineMessageAuditDetails(userQueryMsg, 1), 'query': JSON.stringify(this.query) ,'is-user-input': true, 'forced-workflow': this.config.modeSettings.initialization.forcedWorkflow, 'enabled-functions': this.config.defaultValues.functions?.filter(func => func.enabled).map(func => func.name), 'additional-workflow-properties': JSON.stringify(this.config.additionalWorkflowProperties)});\n      }\n    } else {\n      const warningMsg = {role: 'search-warning', content: this.config.globalSettings.searchWarningMessage, additionalProperties: {display: true}};\n      this.openChat([systemMsg, warningMsg]);\n      this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(warningMsg, 0));\n    }\n  }\n\n  private _defineMessageAuditDetails(message: ChatMessage, rank: number): Record<string, any> {\n    return {\n      'duration': 0,\n      'text': message.content,\n      'role': message.role,\n      'rank': rank\n    };\n  }\n\n  /**\n   * Start/open a new chat with the provided messages and chatId\n   * If the last message is from the user, a request to the assistant is made to get an answer\n   * If the last message is from the assistant, the conversation is loaded right away\n   * @param messages The list of messages of the chat\n   * @param savedChatId  The id of the saved chat. If provided (ie. an existing discussion in the saved chat index), update the savedChatId in the chat service for the upcoming saved chat operations\n   */\n  openChat(messages: RawMessage[], savedChatId?: string) {\n    if (!messages || !Array.isArray(messages)) {\n      console.error('Error occurs while trying to load the chat discussion. Invalid messages received :', messages);\n      return;\n    }\n    if (savedChatId) {\n      this.chatService.setSavedChatId(savedChatId);\n      this.chatService.generateChatId(savedChatId);\n    }\n    this.resetChat();\n    this.messages$.next(messages);\n    this.chatService.chatHistory = messages;\n    const lastMessage = messages.at(-1);\n    if(lastMessage && lastMessage.role === 'user') {\n      this.fetch(messages); // If the last message if from a user, an answer from the assistant is expected\n    }\n    else {\n      this.updateData(messages); // If the last message if from the assistant, we can load the conversation right away\n      this.terminateFetch();\n    }\n    this._addScrollListener();\n  }\n\n  /**\n   * Reset the chat by clearing the chat history and the UI accordingly\n   * The user input will be cleared\n   * The fetch subscription will be terminated\n   */\n  resetChat() {\n    if(this.messages$.value) {\n      this.messages$.next(undefined); // Reset chat\n    }\n    this.chatService.chatHistory = undefined; // Reset chat history\n    this.question = '';\n    this.terminateFetch();\n  }\n\n  onLoadChat() {\n    this.loading$.next(true);\n    this._sub.add(\n      this.chatService.loadSavedChat$\n        .pipe(\n          filter(savedChat => !!savedChat),\n          switchMap(savedChat => this.chatService.getSavedChat(savedChat!.id)),\n          filter(savedChatHistory => !!savedChatHistory),\n          tap(savedChatHistory => this.openChat(savedChatHistory!.history, savedChatHistory!.id))\n        ).subscribe()\n    );\n  }\n\n  stopGeneration() {\n    this.chatService.stopGeneration().subscribe(\n      () => this.terminateFetch()\n    );\n  }\n\n  terminateFetch() {\n    this._dataSubscription?.unsubscribe();\n    this._dataSubscription = undefined;\n    this.loading$.next(false);\n    this.cdr.detectChanges();\n\n    if (this.focusAfterResponse) {\n      setTimeout(() => {\n        this.questionInput?.nativeElement.focus();\n      });\n    }\n  }\n\n  /**\n   * Copy a previous user message of the chat history to the chat user input.\n   * Thus, the user can edit and resubmit the message.\n   * Once the edited message is submitted, all subsequent messages starting from @param index will be removed from the history and the UI will be updated accordingly.\n   * The assistant will regenerate a new answer based on the updated chat history.\n   * @param index The index of the user's message to edit\n   */\n  editMessage(index: number) {\n    this.messageToEdit = index;\n    this.remappedMessageToEdit = this._remapIndexInChatHistory(index);\n    this.question = this.chatService.chatHistory![this._remapIndexInChatHistory(index)].content;\n    this.chatService.generateAuditEvent('edit.click', {'rank': this._remapIndexInChatHistory(index)});\n  }\n\n  /**\n   * Copy a previous assistant message of the chat history to the clipboard.\n   * @param index The index of the assistant's message to edit\n   */\n  copyMessage(index: number) {\n    // Remap the index in the chat history\n    const idx = this._remapIndexInChatHistory(index);\n    this.chatService.generateAuditEvent('copy.click', {'rank': idx});\n  }\n\n  /**\n   * Starting from the provided index, remove all subsequent messages from the chat history and the UI accordingly.\n   * The assistant will regenerate a new answer based on the updated chat history.\n   * @param index The index of the assistant's message to regenerate\n   */\n  regenerateMessage(index: number) {\n    // Update the messages in the UI by removing all subsequent 'assistant' messages starting from the provided index until the first previous 'user' message\n    let i = index;\n    while (i >= 0 && (this.messages$.value!)[i].role !== 'user') {\n      i--;\n    }\n    // It should always be the case that i > 0\n    if (i >= 0) {\n      this.messages$.next(this.messages$.value!.slice(0, i+1));\n      // Remap the index of this found first previous 'user' message in the chat history\n      const idx = this._remapIndexInChatHistory(i);\n      // Define and Update the chat history based on which the assistant will generate a new answer\n      this.chatService.chatHistory = this.chatService.chatHistory!.slice(0, idx+1);\n      // Fetch the answer\n      this.fetch(this.chatService.chatHistory);\n      this.chatService.generateAuditEvent('regenerate.click', {'rank': idx});\n    }\n  }\n\n  /**\n   * Remaps the index in the chat history.\n   * The chat history is a list of messages where some messages can be hidden (display set to false).\n   * The index provided as input is the index of the message in the chat history displayed in the UI.\n   * This function should be removed once the backend is updated to add the ids of the messages in the chat history\n   * @param index - The index to be remapped.\n   */\n  private _remapIndexInChatHistory(index: number) {\n    // a copy of the chat history is created to avoid modifying the original chat history. Additionally, a rank is giving to each message.\n    const history = this.chatService.chatHistory!.slice().map((message, idx) => {\n      return {...message, additionalProperties: {...message.additionalProperties, rank: idx}};\n    });\n    // Count the number of hidden messages in messages$ before the provided index\n    // This is mandatory to get the correct rank of the message in the chat history\n    // Since some hidden messages (like 'system' messages) are not displayed in the UI but have been counted in the provided index\n    const numberOfHiddenMessagesInMessages$BeforeIndex = this.messages$.value!.slice(0, index).filter(message => !message.additionalProperties.display).length;\n    // remove all messages that have display set to false\n    // this is mandatory since at the point of time when the assistant answers a question,\n    // it might have some hidden messages (for example contextMessages) that are available in the chat history but don't figure in messages$ unless a new question is asked\n    const filteredHistory = history.filter(message => message.additionalProperties.display);\n    // return the index of the message in the filtered history\n    return filteredHistory[index - numberOfHiddenMessagesInMessages$BeforeIndex].additionalProperties.rank;\n  }\n\n  onKeyUp(event: KeyboardEvent): void {\n    switch (event.key) {\n      case 'Backspace':\n        this.calculateHeight();\n        break;\n      case 'Enter':\n        if (!event.shiftKey) {\n          event.preventDefault();\n          this.submitQuestion();\n        }\n        this.calculateHeight();\n        break;\n      default:\n        break;\n    }\n  }\n\n  calculateHeight(event?: KeyboardEvent): void {\n    if (event?.key === 'Enter' && !event.shiftKey) {\n      event?.preventDefault();\n    }\n    const maxHeight = 170;\n    const el = this.questionInput!.nativeElement;\n    el.style.maxHeight = `${maxHeight}px`;\n    el.style.height = 'auto';\n    el.style.height = `${el.scrollHeight}px`;\n    el.style.overflowY = el.scrollHeight >= maxHeight ? 'scroll' : 'hidden';\n  }\n\n  /**\n   * Send a \"like\" event on clicking on the thumb-up icon of an assistant's message\n   * @param message The assistant message to like\n   * @param rank The rank of the message to like\n   */\n  onLike(message: ChatMessage, rank: number): void {\n    // Remap the index in the chat history\n    const idx = this._remapIndexInChatHistory(rank);\n    this.chatService.generateAuditEvent('thumb-up.click', {rank: idx});\n    this.reportType = 'like';\n    this.messageToReport = message;\n    this.reportComment = undefined;\n    this.reportRank = rank;\n    this.showReport = true\n  }\n\n\n  /**\n   * Send a \"dislike\" event on clicking on the thumb-down icon of an assistant's message.\n   * It also opens the issue reporting dialog.\n   * @param message The assistant message to dislike\n   * @param index The rank of the message to dislike\n   */\n  onDislike(message: ChatMessage, rank: number): void {\n    // Remap the index in the chat history\n    const idx = this._remapIndexInChatHistory(rank);\n    this.chatService.generateAuditEvent('thumb-down.click', {rank: idx});\n    this.reportType = 'dislike';\n    this.messageToReport = message;\n    this.issueType = '';\n    this.reportComment = undefined;\n    this.reportRank = rank;\n    this.showReport = true\n  }\n\n  /**\n   * Report an issue related to the assistant's message.\n   */\n  sendReport(): void {\n    const details = {\n      'comment': this.reportComment,\n      'text': this.messageToReport!.content,\n      'rank': this.reportRank,\n    };\n    if (this.reportType === 'dislike') {\n      details['report-type'] = this.issueType;\n      this.chatService.generateAuditEvent('negative-report.send', details);\n    } else {\n      this.chatService.generateAuditEvent('positive-report.send', details);\n    }\n    this.showReport = false;\n  }\n\n  /**\n   * Close the reporting dialog.\n   */\n  ignoreReport(): void {\n    this.showReport = false;\n  }\n\n  /**\n   * Handle the click on a reference's 'open preview'.\n   * @param data\n   */\n  openAttachmentPreview(data: {reference: ChatContextAttachment, partId?: number}) {\n    this.openPreview.emit(data.reference);\n    const details = {\n      'doc-id': data.reference.recordId,\n      'title': data.reference.record.title,\n      'source': data.reference.record.treepath,\n      'collection': data.reference.record.collection,\n      'index': data.reference.record.databasealias,\n    };\n    if(!!data.partId) details['part-id'] = data.partId;\n    this.chatService.generateAuditEvent('attachment.preview.click', details);\n  }\n\n  /**\n   * Handle the click on a reference's 'open original document'.\n   * @param data\n   */\n  openOriginalAttachment(data: {reference: ChatContextAttachment, partId?: number}) {\n    this.openDocument.emit(data.reference.record);\n    const details = {\n      'doc-id': data.reference.recordId,\n      'title': data.reference.record.title,\n      'source': data.reference.record.treepath,\n      'collection': data.reference.record.collection,\n      'index': data.reference.record.databasealias,\n    };\n    if(!!data.partId) details['part-id'] = data.partId;\n    this.chatService.generateAuditEvent('attachment.link.click', details);\n  }\n\n  /**\n   * Handle the click on a suggested action.\n   * @param action Suggested action.\n   * @param index Rank of the message in the chatHistory related to the suggested action.\n   */\n  suggestActionClick(action: SuggestedAction, index: number) {\n    this.suggestAction.emit(action);\n    this.chatService.generateAuditEvent('suggestedAction.click', {'text': action.content, 'suggestedAction-type': action.type})\n  }\n\n  /**\n   * Handle the click on a chat starter.\n   * @param starter the chat starter.\n   */\n  chatStarterClick(starter: ChatStarter) {\n    this.chatStarter.emit(starter);\n  }\n\n  /**\n   * It looks for the debug messages available in the current group of \"assistant\" messages.\n   * By design, the debug messages are only available in the first visible message among the group \"assistant\" messages.\n   * @param index The rank of the message\n   * @returns The debug messages available in the current group of \"assistant\" messages\n   */\n  getDebugMessages(index: number): DebugMessage[] {\n    // If it is not an assistant message, return\n    if ((this.messages$.value!)[index].role !== 'assistant') {\n      return [];\n    }\n    // Get the array of messages up to the indicated index\n    const array = this.messages$.value!.slice(0, index + 1);\n    // If it is an assistant message, look for the debug messages available in the current group of \"assistant\" messages\n    // By design, the debug messages are only available in the first visible message among the group \"assistant\" messages.\n    const idx = this.chatService.firstVisibleAssistantMessageIndex(array);\n    if (idx > -1) {\n      return (this.messages$.value!)[idx].additionalProperties.$debug || [];\n    }\n    return [];\n  }\n\n  /**\n   * Handle the click on the 'show log info' button of a message.\n   * @param index The rank of the message\n   */\n  showDebug(index: number): void {\n    this.debugMessages = this.getDebugMessages(index);\n    this.showDebugMessages = true;\n    this.cdr.detectChanges();\n  }\n\n  /**\n   * Verify whether the current message is an assistant message and that all following messages are assistant ones\n   * Used to keep the \"View progress\" opened even though the assistant is sending additional messages after the current one\n   * @param messages the list of current messages\n   * @param index the index of the current message\n   * @returns if this messages and the following ones (if any) are the last ones\n   */\n  isAssistantLastMessages(messages: ChatMessage[], index: number): boolean {\n    for (let i = index; i < messages.length; i++) {\n      if (messages[i].role !== 'assistant') return false;\n    }\n    return true;\n  }\n}\n","<ng-container *ngIf=\"!initializationError\">\n  <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n    <!-- Token consumption -->\n    <div class=\"ms-1\" *ngIf=\"config?.globalSettings?.displayUserQuotaConsumption || config?.globalSettings?.displayChatTokensConsumption\">\n      <ng-container *ngTemplateOutlet=\"tokenConsumptionTpl || defaultTokenConsumptionTpl; context: { $implicit: instanceId }\"></ng-container>\n    </div>\n\n    <!-- Chat Messages -->\n    <ul class=\"list-group list-group-flush overflow-auto flex-grow-1 pe-2 pb-2\" #messageList>\n      <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n        <!-- Regular messages -->\n        <li class=\"list-group-item\"\n          *ngIf=\"message.additionalProperties.display\"\n          [style.--bs-list-group-item-padding-y.rem]=\"'0.6'\"\n          [class.opacity-50]=\"messageToEdit && messageToEdit < index + 1\">\n          <sq-chat-message\n            [class.sq-user-message]=\"message.role === 'user'\"\n            [class.last-message]=\"last\"\n            [message]=\"message\"\n            [conversation]=\"messages\"\n            [suggestedActions]=\"last ? message.additionalProperties.$suggestedAction : undefined\"\n            [chatStarters]=\"visibleMessagesCount === 1 ? config.modeSettings.initialization.chatStarters : undefined\"\n            [assistantMessageIcon]=\"assistantMessageIcon\"\n            [userMessageIcon]=\"userMessageIcon\"\n            [connectionErrorMessageIcon]=\"connectionErrorMessageIcon\"\n            [searchWarningMessageIcon]=\"searchWarningMessageIcon\"\n            [streaming]=\"(chatService.streaming$ | async) && (last || isAssistantLastMessages(messages, index))\"\n            [canEdit]=\"(loading$ | async) === false && ((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role === 'user'\"\n            [canCopy]=\"((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role !== 'connection-error' && message.role !== 'search-warning'\"\n            [canLike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n            [canDislike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n            [canDebug]=\"(((chatService.streaming$ | async) === false && last) || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && isAdmin && (getDebugMessages(index).length > 0) && config?.defaultValues.debug\"\n            [canRegenerate]=\"(loading$ | async) === false && (((chatService.streaming$ | async) === false  && last) || (!last && messages[index+1].role !== 'assistant'))  && message.role === 'assistant' && messageToEdit === undefined\"\n            (edit)=\"editMessage(index)\"\n            (copy)=\"copyMessage(index)\"\n            (regenerate)=\"regenerateMessage(index)\"\n            (openDocument)=\"openOriginalAttachment($event)\"\n            (openPreview)=\"openAttachmentPreview($event)\"\n            (suggestAction)=\"suggestActionClick($event, index)\"\n            (chatStarterClicked)=\"chatStarterClick($event)\"\n            (like)=\"onLike($event, index)\"\n            (dislike)=\"onDislike($event, index)\"\n            (debug)=\"showDebug(index)\">\n          </sq-chat-message>\n        </li>\n      </ng-container>\n      <!-- Loading spinner -->\n      <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n        <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n      </li>\n    </ul>\n\n    <!-- Reporting a feedback form -->\n    <div class=\"issue-report bg-light pt-3 pb-2\" *ngIf=\"showReport\">\n      <ng-container *ngTemplateOutlet=\"reportTpl || reportTplDefault; context: { $implicit: messageToReport, rank: reportRank, type: reportType }\"></ng-container>\n    </div>\n\n    <!-- User text input -->\n    <div class=\"user-input mt-auto\" *ngIf=\"!showReport\">\n      <div class=\"py-2\">\n        <div [hidden]=\"!isConnected\">\n          <ng-container *ngIf=\"enabledUserInput\" [ngTemplateOutlet]=\"inputTpl\"></ng-container>\n        </div>\n        <!-- Retry button -->\n        <button [hidden]=\"isConnected\" class=\"btn mb-4 ast-error ast-btn sq-retry\" (click)=\"retryFetch()\">\n          <span>Try again</span>\n          <span *ngIf=\"retrialAttempts\" class=\"ms-2 attempts\">{{ retrialAttempts }}</span>\n        </button>\n        <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n          {{ config?.globalSettings?.disclaimer }}\n        </div>\n      </div>\n    </div>\n\n    <!-- Floating scroll button -->\n    <div *ngIf=\"!isAtBottom && !showReport\" class=\"sq-floating-scroll\" [ngClass]=\"enabledUserInput ? 'sq-floating-scroll--when-user-input' : 'sq-floating-scroll--without-user-input'\">\n      <button class=\"btn shadow\" (click)=\"scrollDown()\">\n        <i class=\"fas fa-angle-double-down\"></i>\n      </button>\n    </div>\n  </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n  <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n    <span class=\"visually-hidden\">Loading...</span>\n  </div>\n</ng-template>\n\n<ng-template #inputTpl>\n  <div class=\"px-3 py-1\">\n    <div class=\"ast-input-container\">\n      <button disabled class=\"btn btn-light\">\n        <i class=\"fas fa-search\"></i>\n      </button>\n      <textarea #questionInput rows=\"1\"\n        type=\"text\" class=\"form-control\"\n        placeholder=\"Ask something\" autofocus\n        [(ngModel)]=\"question\"\n        (keyup)=\"onKeyUp($event)\"\n        (keydown)=\"calculateHeight($event)\"\n        [disabled]=\"(loading$ | async) || (chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\">\n      </textarea>\n      <button\n        *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async) && !(chatService.stoppingGeneration$ | async)\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        sqTooltip=\"Send message\"\n        (click)=\"submitQuestion()\">\n        <i class=\"fas fa-paper-plane\"></i>\n      </button>\n      <button\n        *ngIf=\"messageToEdit\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        sqTooltip=\"Cancel edition\"\n        (click)=\"messageToEdit = undefined; question = ''\">\n        <i class=\"fas fa-undo-alt\"></i>\n      </button>\n      <span *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n        Generating <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n      </span>\n      <span *ngIf=\"(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n        Stopping <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n      </span>\n      <button\n        *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        sqTooltip=\"Stop generating\"\n        (click)=\"stopGeneration()\">\n        <i class=\"fas fa-stop\"></i>\n      </button>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #reportTplDefault let-message let-rank=\"rank\" let-type=\"type\">\n  <div class=\"px-3\">\n    <ng-container *ngIf=\"type === 'dislike'\">\n      <h5>Issue type</h5>\n      <select class=\"form-select mb-4\" [(ngModel)]=\"issueType\">\n        <option [value]=\"''\">Choose an issue type</option>\n        <option *ngFor=\"let type of (issueTypes ?? defaultIssueTypes)\">{{type}}</option>\n      </select>\n      <h5>What was unsatisfying about this response? (optional)</h5>\n    </ng-container>\n    <ng-container *ngIf=\"type === 'like'\">\n      <h5>Why did you like this answer? (optional)</h5>\n    </ng-container>\n    <textarea class=\"form-control\" [(ngModel)]=\"reportComment\" placeholder=\"Write your comment\"></textarea>\n    <div class=\"d-flex flex-row-reverse mt-2\">\n      <button class=\"btn btn-primary\" [disabled]=\"type === 'dislike' && !issueType\" (click)=\"sendReport()\">Send</button>\n      <button class=\"btn btn-light\" (click)=\"ignoreReport()\">Cancel</button>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #defaultTokenConsumptionTpl let-instanceId>\n  <sq-token-progress-bar\n    [instanceId]=\"instanceId\">\n  </sq-token-progress-bar>\n</ng-template>\n\n<div class=\"debug-messages\" [class.displayed]=\"showDebugMessages\">\n  <button *ngIf=\"showDebugMessages\" class=\"btn btn-light shadow back-btn\" (click)=\"showDebugMessages=false\">\n    <i class=\"fas fa-chevron-right\"></i>\n  </button>\n  <ng-container *ngTemplateOutlet=\"debugMessagesTpl || defaultDebugMessagesTpl; context: { $implicit: debugMessages }\">\n  </ng-container>\n</div>\n\n<ng-template #defaultDebugMessagesTpl let-debugMessages>\n  <sq-debug-message [data]=\"debugMessages\"></sq-debug-message>\n</ng-template>\n"]}
|
|
1057
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat.component.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/chat.component.ts","../../../../projects/assistant/chat/chat.component.html"],"names":[],"mappings":"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","sourcesContent":["import { inject, ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, ElementRef, EventEmitter, Input, OnChanges, OnDestroy, OnInit, Output, SimpleChanges, TemplateRef, ViewChild } from \"@angular/core\";\nimport { Action } from \"@sinequa/components/action\";\nimport { AbstractFacet } from \"@sinequa/components/facet\";\nimport { SearchService } from \"@sinequa/components/search\";\nimport { AppService, Query } from \"@sinequa/core/app-utils\";\nimport { PrincipalWebService, Record as Article } from \"@sinequa/core/web-services\";\nimport { BehaviorSubject, Subscription, combineLatest, filter, fromEvent, map, merge, switchMap, take, tap } from \"rxjs\";\nimport { ChatService } from \"./chat.service\";\nimport { ChatContextAttachment, ChatConfig, ChatMessage, GllmModelDescription, MessageHandler, RawMessage, SuggestedAction, InitChat, DebugMessage } from \"./types\";\nimport { InstanceManagerService } from \"./instance-manager.service\";\nimport { WebSocketChatService } from \"./websocket-chat.service\";\nimport { ChatMessageComponent } from \"./chat-message/chat-message.component\";\nimport { CommonModule } from \"@angular/common\";\nimport { FormsModule } from \"@angular/forms\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { RestChatService } from \"./rest-chat.service\";\nimport { TokenProgressBarComponent } from \"./token-progress-bar/token-progress-bar.component\";\nimport { DebugMessageComponent } from \"./debug-message/debug-message.component\";\nimport { HubConnection, HubConnectionState  } from \"@microsoft/signalr\";\nimport { UtilsModule } from \"@sinequa/components/utils\";\nimport { NotificationsService } from \"@sinequa/core/notification\";\n\n@Component({\n  selector: 'sq-chat-v3', // mandatory since @sinequa/components already has the same tag-name \"sq-chat\"\n  templateUrl: './chat.component.html',\n  styleUrls: ['./chat.component.scss'],\n  providers: [\n    RestChatService,\n    WebSocketChatService\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true,\n  imports: [CommonModule, FormsModule, ChatMessageComponent, TokenProgressBarComponent, DebugMessageComponent, UtilsModule]\n})\nexport class ChatComponent extends AbstractFacet implements OnInit, OnChanges, OnDestroy {\n\n  public loginService = inject(LoginService);\n  public websocketService = inject(WebSocketChatService);\n  public restService = inject(RestChatService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public searchService = inject(SearchService);\n  public principalService = inject(PrincipalWebService);\n  public cdr = inject(ChangeDetectorRef);\n  public appService = inject(AppService);\n  public notificationsService = inject(NotificationsService);\n\n  /** Define the key based on it, the chat service instance will be stored */\n  @Input() instanceId: string;\n  /** Define the query to use to fetch answers */\n  @Input() query: Query = this.searchService.query;\n  /** Function that determines whether the chat should be reloaded after the query changes\n   * If not provided, the chat will be reloaded by default\n   * @param prevQuery The previous query\n   * @param newQuery The new query\n   * @returns true if the chat should be reloaded, false otherwise\n   */\n  @Input() queryChangeShouldTriggerReload: (prevQuery: Query, newQuery: Query) => boolean;\n  /** Define the protocol to be used for this chat instance*/\n  @Input() protocol: 'REST' | 'WEBSOCKET' = \"WEBSOCKET\";\n  /** Map of listeners overriding default registered ones*/\n  @Input() messageHandlers: Map<string, MessageHandler<any>> = new Map();\n  /** When the assistant answer a user question, automatically scroll down to the bottom of the discussion */\n  @Input() automaticScrollToLastResponse = false;\n  /** When the assistant answer a user question, automatically focus to the chat input */\n  @Input() focusAfterResponse = false;\n  /** A chat discussion that the component should get initialized with it */\n  @Input() chat?: InitChat;\n  /** Icon to use for the assistant messages */\n  @Input() assistantMessageIcon = 'sq-sinequa';\n  /** Icon to use for the user messages */\n  @Input() userMessageIcon: string;\n  /** Icon to use for the connection error messages */\n  @Input() connectionErrorMessageIcon: string;\n  /** Icon to use for the search warning messages */\n  @Input() searchWarningMessageIcon: string;\n  /** Event emitter triggered once the signalR connection is established  */\n  @Output() connection = new EventEmitter<HubConnection>();\n  /** Event emitter triggered each time the assistant updates the current chat */\n  /** Event emitter triggered when the chat is loading new content */\n  @Output(\"loading\") loading$ = new EventEmitter<boolean>(false);\n  /** Emits the assistant configuration used when instantiating the component */\n  @Output(\"config\") _config = new EventEmitter<ChatConfig>();\n  @Output() data = new EventEmitter<ChatMessage[]>();\n  /** Event emitter triggered when the user clicks to open the original document representing the context attachment*/\n  @Output() openDocument = new EventEmitter<Article>();\n  /** Event emitter triggered when the user clicks to open the preview of a document representing the context attachment */\n  @Output() openPreview = new EventEmitter<ChatContextAttachment>();\n  /** Event emitter triggered when the user clicks on a suggested action */\n  @Output() suggestAction = new EventEmitter<SuggestedAction>();\n  /** ViewChild decorators to access the template elements */\n  @ViewChild('messageList') messageList?: ElementRef<HTMLUListElement>;\n  @ViewChild('questionInput') questionInput?: ElementRef<HTMLTextAreaElement>;\n  /** ContentChild decorators allowing the override of the default templates from the parent component */\n  @ContentChild('loadingTpl') loadingTpl?: TemplateRef<any>;\n  @ContentChild('reportTpl') reportTpl?: TemplateRef<any>;\n  @ContentChild('tokenConsumptionTpl') tokenConsumptionTpl?: TemplateRef<any>;\n  @ContentChild('debugMessagesTpl') debugMessagesTpl?: TemplateRef<any>;\n\n  chatService: ChatService;\n  config: ChatConfig;\n  messages$ = new BehaviorSubject<ChatMessage[] | undefined>(undefined);\n\n  question = '';\n\n  _actions: Action[] = [];\n  private _resetChatAction = new Action({\n    icon: 'fas fa-sync',\n    title: \"Reset assistant\",\n    action: () => this.newChat()\n  });\n\n  private _sub = new Subscription();\n  private _dataSubscription: Subscription | undefined;\n\n  /** Variables that depend on the type of model in use */\n  modelDescription?: GllmModelDescription;\n\n  messageToEdit?: number;\n  remappedMessageToEdit?: number;\n  changes$ = new BehaviorSubject<SimpleChanges | undefined>(undefined);\n  currentMessageIndex: number | undefined;\n  firstChangesHandled = false;\n  isAtBottom = true;\n  initializationError = false;\n  enabledUserInput = false;\n  isConnected = true; // By default, the chat is considered connected\n  retrialAttempts: number | undefined;\n  // Flag to track whether the 'reconnected' listener is already registered\n  private _isReconnectedListenerRegistered = false;\n\n  // Issue reporting\n  issueTypes?: string[];\n  defaultIssueTypes: string[] = [\n    'User Interface bug',\n    'Incorrect or misleading response',\n    'Incomplete response',\n    'Technical issue',\n    'Privacy/data security issue',\n    'Other'\n  ];\n  issueType: string = '';\n  reportComment?: string;\n  messageToReport?: ChatMessage;\n  reportRank?: number;\n  reportType: 'like' | 'dislike' = 'dislike';\n  showReport = false;\n\n  // Debug messages\n  debugMessages: DebugMessage[] | undefined;\n  showDebugMessages = false;\n\n  private _previousQuery: Query;\n  private _reloadSubscription: Subscription | undefined = undefined;\n\n  constructor() {\n    super();\n    this._actions.push(this._resetChatAction);\n  }\n\n  ngOnInit(): void {\n    this._sub.add(\n      this.loginService.events.pipe(\n        filter(e => e.type === 'login-complete'),\n        tap(_ => this.instantiateChatService()),\n        map(_ => this.chatService.initChatConfig()),\n        switchMap(() => this.chatService.initConfig$),\n        filter(initConfig => !!initConfig),\n        switchMap(_ => this.chatService.init()),\n        switchMap(_ => this.chatService.initProcess$),\n        filter(success => !!success),\n        tap(_ => {\n          if (this.chatService instanceof WebSocketChatService) {\n            this.connection.emit(this.chatService.connection);\n          }\n          this.onLoadChat();\n        }),\n        tap(_ => this.chatService.overrideUser()),\n        switchMap(_ => this.chatService.userOverride$),\n        switchMap(_ => this.chatService.assistantConfig$),\n        tap(config => {\n          this.config = config!;\n          this.enabledUserInput = this.config.modeSettings.enabledUserInput;\n          this.issueTypes = this.config.auditSettings?.issueTypes?.length ? this.config.auditSettings!.issueTypes : undefined;\n          this._config.emit(config);\n          try {\n            this.updateModelDescription();\n            if(!this.firstChangesHandled) {\n              this._previousQuery = JSON.parse(JSON.stringify(this.query)); // Initialize the previous query\n              this._handleChanges();\n              this._addScrollListener();\n              this.firstChangesHandled = true;\n            }\n          } catch (error) {\n            this.initializationError = true\n            throw error;\n          }\n        })\n      ).subscribe()\n    );\n\n    this._sub.add(\n      combineLatest([\n        this.chatService.streaming$,\n        this.chatService.stoppingGeneration$\n      ]).pipe(\n        map(([streaming, stoppingGeneration]) => !!(streaming || stoppingGeneration))\n      ).subscribe((result) => {\n        this._resetChatAction.disabled = result;\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    this.changes$.next(changes);\n    if (this.config) {\n      this._handleChanges();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this._sub.unsubscribe();\n    this._dataSubscription?.unsubscribe();\n    this._reloadSubscription?.unsubscribe();\n    if (this.chatService instanceof WebSocketChatService) {\n      this.chatService.stopConnection();\n    }\n  }\n\n  get isAdmin(): boolean {\n    return this.principalService.principal?.isAdministrator || false;\n  }\n\n  /**\n   * Instantiate the chat service based on the provided @input protocol\n   * This chat service instance will then be stored in the instanceManagerService with provided @input instanceId as a key\n   */\n  instantiateChatService(): void {\n    switch (this.protocol) {\n      case 'REST':\n        this.chatService = this.restService;\n        break;\n      case 'WEBSOCKET':\n        this.chatService = this.websocketService;\n        break;\n      default:\n        throw new Error(`Could not found a ChatService implementation corresponding to the provided protocol: '${this.protocol}'`);\n    }\n    this.chatService.setChatInstanceId(this.instanceId);\n    this.instanceManagerService.storeInstance(this.instanceId, this.chatService);\n  }\n\n  override get actions() { return this._actions; }\n\n\n  /**\n   * Handles the changes in the chat component.\n   * If the chat service is a WebSocketChatService, it handles the override of the message handlers if they exist.\n   * Initializes the chat with the provided chat messages if they exist, otherwise loads the default chat.\n   * If the chat is initialized, the initialization event is \"Query\", the query changes, and the queryChangeShouldTriggerReload function is provided,\n   * then the chat should be reloaded if the function returns true. Otherwise, the chat should be reloaded by default.\n   * It takes into account the ongoing streaming process and the ongoing stopping process to trigger that conditionally define the logic\n   * of the reload :\n   * - If the chat is streaming, then stop the generation and wait for the fetch to complete before reloading the chat.\n   * - If the chat is stopping the generation, then wait for the fetch to complete before reloading the chat.\n   */\n  private _handleChanges() {\n    const changes = this.changes$.value;\n    // If the chat service is a WebSocketChatService, handle the override of the message handlers if exists\n    if (changes?.messageHandlers && this.messageHandlers && this.chatService instanceof WebSocketChatService) {\n      this.chatService.overrideMessageHandlers(this.messageHandlers);\n    }\n    /**\n     * Initialize the chat with the provided chat messages if exists, otherwise load the default chat\n     * Once the chat is initialized (firstChangesHandled is true), allow opening the chat with the new provided messages (if exists)\n     */\n    if (!this.firstChangesHandled || changes?.chat) {\n      const openChat = () => {\n        if (this.messages$.value) {\n          this.chatService.listSavedChat(); // Refresh the list of saved chats\n        }\n        this.openChat(this.chat!.messages);\n      };\n      this.chatService.generateChatId();\n      if (this.chat) {\n        this.chatService.generateAuditEvent('new-chat', {'configuration': JSON.stringify(this.chatService.assistantConfig$.value),'chat-init': JSON.stringify(this.chat)});\n        openChat();\n      } else {\n        this.chatService.generateAuditEvent('new-chat', {'configuration': JSON.stringify(this.chatService.assistantConfig$.value)});\n        this.loadDefaultChat();\n      }\n    }\n    /**\n     * If the chat is initialized, the initialization event is \"Query\", the query changes and the queryChangeShouldTriggerReload function is provided,\n     * then the chat should be reloaded if the function returns true\n     * Otherwise, the chat should be reloaded by default\n     */\n    if (this.firstChangesHandled && changes?.query && this.config.modeSettings.initialization.event === 'Query') {\n      if (this.queryChangeShouldTriggerReload ? this.queryChangeShouldTriggerReload(this._previousQuery, this.query) : true) {\n        if (!!this.chatService.stoppingGeneration$.value) {\n          if (!this._reloadSubscription) {\n            // Create a subscription to wait for both streaming$ and stoppingGeneration$ to be false\n            this._reloadSubscription = combineLatest([\n              this.chatService.streaming$,\n              this.chatService.stoppingGeneration$\n            ])\n            .pipe(\n              filter(([streaming, stopping]) => !streaming && !stopping), // Wait until both are false\n              take(1) // Complete after the first match\n            ).subscribe(() => {\n              // Execute the reload after the query change\n              this._triggerReloadAfterQueryChange();\n              // Update _previousQuery with the current query\n              this._previousQuery = JSON.parse(JSON.stringify(this.query));\n              // Clean up subscription and reset its value\n              this._reloadSubscription = undefined;\n            });\n          }\n        } else if (!!this.chatService.streaming$.value) {\n          if (!this._reloadSubscription) {\n            this._reloadSubscription = this.chatService.stopGeneration()\n            .subscribe({\n              next: () => {},\n              error: () => {\n                // Clean up subscription and reset its value\n                this._reloadSubscription?.unsubscribe();\n                this._reloadSubscription = undefined;\n              },\n              complete: () => {\n                // Wait for the ongoing fetch to complete, then trigger the reload\n                this.chatService.streaming$.pipe(\n                  filter((streaming) => !streaming),\n                  take(1)\n                ).subscribe(() => {\n                  // Execute the reload after the query change\n                  this._triggerReloadAfterQueryChange();\n                  // Update _previousQuery with the current query\n                  this._previousQuery = JSON.parse(JSON.stringify(this.query));\n                  // Clean up subscription and reset its value\n                  this._reloadSubscription!.unsubscribe();\n                  this._reloadSubscription = undefined;\n                });\n              }\n            });\n          }\n        } else {\n          // Execute the reload after the query change\n          this._triggerReloadAfterQueryChange();\n          // Update _previousQuery with the current query\n          this._previousQuery = JSON.parse(JSON.stringify(this.query));\n        }\n      } else {\n        // Update _previousQuery with the current query\n        this._previousQuery = JSON.parse(JSON.stringify(this.query));\n      }\n    }\n  }\n\n  /**\n   * Triggers a reload after the query change.\n   * This method performs the necessary operations to reload the chat after a query change.\n   * It sets the system and user messages, resets the savedChatId, generates a new chatId,\n   * generates a new chat audit event, and handles the query mode.\n   */\n  private _triggerReloadAfterQueryChange() {\n    const systemMsg = {role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: {display: false}};\n    const userMsg = {role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, {principal: this.principalService.principal}), additionalProperties: {display: this.config.modeSettings.displayUserPrompt}};\n    this.chatService.setSavedChatId(undefined); // Reset the savedChatId\n    this.chatService.generateChatId(); // Generate a new chatId\n    this.chatService.generateAuditEvent('new-chat', {'configuration': JSON.stringify(this.chatService.assistantConfig$.value)}); // Generate a new chat audit event\n    this._handleQueryMode(systemMsg, userMsg);\n  }\n\n\n  /**\n   * Adds a scroll listener to the message list element.\n   * The listener is triggered when any of the following events occur:\n   * - Loading state changes\n   * - Messages change\n   * - Streaming state changes\n   * - Scroll event occurs on the message list element\n   *\n   * When the listener is triggered, it updates the `isAtBottom` property.\n   */\n  private _addScrollListener() {\n    this._sub.add(\n      merge(this.loading$, this.messages$, this.chatService.streaming$, fromEvent(this.messageList!.nativeElement, 'scroll')).subscribe(() => {\n        this.isAtBottom = this._toggleScrollButtonVisibility();\n        this.cdr.detectChanges();\n      })\n    );\n  }\n\n  /**\n   * Get the model description based on the defaultValues service_id and model_id\n   */\n  updateModelDescription() {\n    this.modelDescription = this.chatService.getModel(this.config.defaultValues.service_id, this.config.defaultValues.model_id);\n    this.cdr.detectChanges();\n  }\n\n  /**\n   * Submits a question from the user.\n   * If the user is editing a previous message, removes all subsequent messages from the chat history.\n   * Triggers the fetch of the answer for the submitted question by calling _fetchAnswer().\n   * Clears the input value in the UI.\n   * ⚠️ If the chat is streaming or stopping the generation, the operation is not allowed.\n   */\n  submitQuestion() {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    if(this.question.trim() && this.messages$.value && this.chatService.chatHistory) {\n      // When the user submits a question, if the user is editing a previous message, remove all subsequent messages from the chat history\n      if (this.messageToEdit !== undefined) {\n        // Update the messages in the UI\n        this.messages$.next(this.messages$.value.slice(0, this.messageToEdit));\n        // Update the raw messages in the chat history which is the clean version used to make the next request\n        this.chatService.chatHistory = this.chatService.chatHistory.slice(0, this.remappedMessageToEdit);\n        this.messageToEdit = undefined;\n        this.remappedMessageToEdit = undefined;\n      }\n      // Remove the search warning message if exists\n      if (this.chatService.chatHistory.at(-1)?.role === 'search-warning') {\n        this.chatService.chatHistory.pop();\n      }\n      // Fetch the answer\n      this._fetchAnswer(this.question.trim(), this.chatService.chatHistory);\n      // Clear the input value in the UI\n      this.questionInput!.nativeElement.value = '';\n      this.questionInput!.nativeElement.style.height = `auto`;\n    }\n  }\n\n  /**\n   * Triggers the fetch of the answer for the given question and updates the conversation.\n   * Generates an audit event for the user input.\n   *\n   * @param question - The question asked by the user.\n   * @param conversation - The current conversation messages.\n   */\n  private _fetchAnswer(question: string, conversation: ChatMessage[]) {\n    const userMsg = {role: 'user', content: question, additionalProperties: {display: true, isUserInput: true, additionalWorkflowProperties: this.config.additionalWorkflowProperties}};\n    const messages = [...conversation, userMsg];\n    this.messages$.next(messages);\n    this.fetch(messages);\n    this.chatService.generateAuditEvent('message', {...this._defineMessageAuditDetails(userMsg, messages.length - 1), 'query': JSON.stringify(this.query), 'is-user-input': true, 'enabled-functions': this.config.defaultValues.functions?.filter(func => func.enabled).map(func => func.name), 'additional-workflow-properties': JSON.stringify(this.config.additionalWorkflowProperties)});\n  }\n\n  /**\n   * Depending on the connection's state :\n   *  - If connected => given a list of messages, the chat endpoint is invoked for a continuation and updates the list of messages accordingly.\n   *  - If any other state => a connection error message is displayed in the chat.\n   * ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.\n   * @param messages The list of messages to invoke the chat endpoint with\n   */\n  public fetch(messages: ChatMessage[]) {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this._updateConnectionStatus();\n    this.cdr.detectChanges();\n\n    if (this.isConnected) {\n      this.loading$.next(true);\n      this._dataSubscription?.unsubscribe();\n      this._dataSubscription = this.chatService.fetch(messages, this.query)\n        .subscribe({\n          next: res => this.updateData(res.history),\n          error: () => {\n            this._updateConnectionStatus();\n            if (!this.isConnected) {\n              const message = {role: 'connection-error', content: this.config.connectionSettings.connectionErrorMessage, additionalProperties: {display: true}};\n              this.messages$.next([...messages, message]);\n            }\n            this.terminateFetch();\n          },\n          complete: () => {\n            // Remove the last message if it's an empty message\n            // This is due to the manner in which the chat service handles consecutive messages\n            const lastMessage = this.messages$.value?.at(-1);\n            if (this.isEmptyAssistantMessage(lastMessage)) {\n              this.messages$.next(this.messages$.value?.slice(0, -1));\n            }\n            this.terminateFetch();\n          }\n        });\n    } else {\n      const message = {role: 'connection-error', content: this.config.connectionSettings.connectionErrorMessage, additionalProperties: {display: true}};\n      this.messages$.next([...messages, message]);\n    }\n\n    if(this.automaticScrollToLastResponse) {\n      this.scrollDown();\n    }\n  }\n\n  /**\n   * Retry to fetch the messages if the connection issues.\n   *  - If reconnecting => keep display the connection error message even when clicking on the \"retry\" button and increasing the number of retrial attempts, until the connection is re-established\n   *  - If disconnected => On click on the \"retry\" button, start the connection process while displaying the connection error message :\n   *      * If successful => given a list of messages, the chat endpoint is invoked for a continuation and updates the list of messages accordingly.\n   *      * If failed => increase the number of retrial attempts\n   */\n  retryFetch() {\n    if(this.chatService instanceof WebSocketChatService) {\n      // A one-time listener for reconnected event\n      const onReconnectedHandler = () => {\n        // Get the messages without the last one (the connection error message)\n        const messages = this.messages$.value!.slice(0, -1);\n        // Find the last \"user\" message in the messages list\n        let index = messages.length - 1;\n        while (index >= 0 && messages[index].role !== 'user') {\n          index--;\n        }\n        // If a user message is found (and it should always be the case), remove all subsequent messages from the chat history\n        // Update the messages in the UI\n        // and fetch the answer from the assistant\n        if (index >= 0) {\n          this.messages$.next(this.messages$.value!.slice(0, index+1));\n          const remappedIndex = this._remapIndexInChatHistory(index);\n          this.chatService.chatHistory = this.chatService.chatHistory!.slice(0, remappedIndex+1);\n          this.fetch(this.chatService.chatHistory);\n        }\n        this.retrialAttempts = undefined; // Reset the number of retrial attempts\n        /**\n         * To remove the handler for onreconnected() after it's been registered,cannot directly use off() like you would for normal events registered with connection.on().\n         * Instead, you need to explicitly remove or reset the handler by assigning it to null or an empty function\n         */\n        (this.chatService as WebSocketChatService).connection!.onreconnected(() => {});\n        // Reset the flag to ensure the handler is registered again when needed\n        this._isReconnectedListenerRegistered = false;\n      }\n      // Depending on the connection's state, take the appropriate action\n      switch (this.chatService.connection!.state) {\n        case HubConnectionState.Connected:\n          // If the connection is re-established in the meantime, fetch the messages\n          onReconnectedHandler();\n          break;\n        case HubConnectionState.Reconnecting:\n          // Attach the reconnected listener if not already registered\n          if (!this._isReconnectedListenerRegistered) {\n            this.chatService.connection!.onreconnected(onReconnectedHandler);\n            this._isReconnectedListenerRegistered = true;\n          }\n          // Increase the number of retrial attempts\n          this.retrialAttempts = this.retrialAttempts ? this.retrialAttempts+1 : 1;\n          break;\n        case HubConnectionState.Disconnected:\n          // Start the new connection\n          this.chatService.startConnection()\n          .then(() => onReconnectedHandler())\n          .catch(() => { // If the connection fails, increase the number of retrial attempts\n            this.retrialAttempts = this.retrialAttempts ? this.retrialAttempts+1 : 1;\n          });\n          break;\n        default:\n          break;\n      }\n    }\n  }\n\n  /**\n   * Check if the signalR connection is connected.\n   * For the REST protocol, the connection is always considered connected (for the moment).\n   */\n  private _updateConnectionStatus() {\n    this.isConnected = (this.chatService instanceof WebSocketChatService) ? this.chatService.connection!.state === HubConnectionState.Connected : true;\n  }\n\n  /**\n   * Update the UI with the new messages\n   * @param messages\n   */\n  updateData(messages: ChatMessage[]) {\n    this.messages$.next(messages);\n    this.data.emit(messages);\n    this.loading$.next(false);\n    this.question = '';\n    if(this.automaticScrollToLastResponse) {\n      this.scrollDown();\n    }\n  }\n\n  /**\n   * @returns true if the chat discussion is scrolled down to the bottom, false otherwise\n   */\n  private _toggleScrollButtonVisibility(): boolean {\n    if(this.messageList?.nativeElement) {\n      return Math.round(this.messageList?.nativeElement.scrollHeight - this.messageList?.nativeElement.scrollTop - 1) <= this.messageList?.nativeElement.clientHeight;\n    }\n    return true;\n  }\n\n  /**\n   * Scroll down to the bottom of the chat discussion\n   */\n  scrollDown() {\n    setTimeout(() => {\n      if(this.messageList?.nativeElement) {\n        this.messageList.nativeElement.scrollTop = this.messageList.nativeElement.scrollHeight;\n        this.cdr.detectChanges();\n      }\n    }, 10);\n  }\n\n  /**\n   * Start a new chat with the defaultValues settings.\n   * The savedChatId in the chat service will be reset, so that the upcoming saved chat operations will be performed on the fresh new chat.\n   * If the savedChat feature is enabled, the list of saved chats will be refreshed.\n   * ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.\n   */\n  newChat() {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.chatService.setSavedChatId(undefined); // Reset the savedChatId\n    this.chatService.generateChatId(); // Generate a new chatId\n    this.chatService.listSavedChat(); // Refresh the list of saved chats\n    this.chatService.generateAuditEvent('new-chat', {'configuration': JSON.stringify(this.chatService.assistantConfig$.value)}); // Generate a new chat audit event\n    this.loadDefaultChat(); // Start a new chat\n  }\n\n  /**\n   * Attaches the specified document IDs to the assistant.\n   * If no document IDs are provided, the operation is not allowed.\n   * If the action for attaching a document is not defined at the application customization level, an error is logged.\n   * ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.\n   * @param ids - An array of document IDs to attach.\n   */\n  attachToChat(ids: string[]) {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    if (!ids || ids?.length < 1) {\n      return;\n    }\n    const attachDocAction = this.config.modeSettings.actions?.[\"attachDocAction\"];\n    if (!attachDocAction) {\n      console.error(`No action is defined for attaching a document to the assistant \"${this.instanceId}\"`);\n      return;\n    }\n    const userMsg = { role: 'user', content: '', additionalProperties: {display: false, isUserInput: false, type: \"Action\", forcedWorkflow: attachDocAction.forcedWorkflow, forcedWorkflowProperties: {...(attachDocAction.forcedWorkflowProperties || {}), ids}, additionalWorkflowProperties: this.config.additionalWorkflowProperties}};\n    const messages = [...this.chatService.chatHistory!, userMsg];\n    this.messages$.next(messages);\n    this.fetch(messages);\n  }\n\n  /**\n   * Start the default chat with the defaultValues settings\n   * If the chat is meant to be initialized with event === \"Query\", the corresponding user query message will be added to the chat history\n   */\n  loadDefaultChat() {\n    // Define the default system prompt and user prompt messages\n    const systemMsg = {role: 'system', content: this.config.defaultValues.systemPrompt, additionalProperties: {display: false}};\n    const userMsg = {role: 'user', content: ChatService.formatPrompt(this.config.defaultValues.userPrompt, {principal: this.principalService.principal}), additionalProperties: {display: this.config.modeSettings.displayUserPrompt}};\n\n    if (this.config.modeSettings.initialization.event === 'Query') {\n      this._handleQueryMode(systemMsg, userMsg);\n    } else {\n      this._handlePromptMode(systemMsg, userMsg);\n    }\n  }\n\n  /**\n   * Handles the prompt mode of the chat component.\n   * If `sendUserPrompt` is true, it opens the chat with both system and user messages,\n   * and generates audit events for both messages.\n   * If `sendUserPrompt` is false, it opens the chat with only the system message,\n   * and generates an audit event for the system message.\n   *\n   * @param systemMsg - The system message to be displayed in the chat.\n   * @param userMsg - The user message to be displayed in the chat (optional).\n   */\n  private _handlePromptMode(systemMsg: ChatMessage, userMsg: ChatMessage) {\n    if (this.config.modeSettings.sendUserPrompt) {\n      this.openChat([systemMsg, userMsg]);\n      this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));\n      this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(userMsg, 1));\n    } else {\n      this.openChat([systemMsg]);\n      this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));\n    }\n  }\n\n  /**\n   * Handles the query mode by displaying the system message, user message, and user query message.\n   * If the provided query text is not empty, then add the user query message to the chat history and invoke the assistant\n   * Otherwise, just start a new chat with a warning message inviting the user to perform a full text search to retrieve some results\n   * @param systemMsg - The system message to be displayed.\n   * @param userMsg - The user message to be displayed.\n   */\n  private _handleQueryMode(systemMsg: ChatMessage, userMsg: ChatMessage) {\n    if (!!this.query.text) {\n      const userQueryMsg = {role: 'user', content: this.query.text, additionalProperties: {display: this.config.modeSettings.initialization.displayUserQuery, query: this.query, forcedWorkflow: this.config.modeSettings.initialization.forcedWorkflow, isUserInput: true, additionalWorkflowProperties: this.config.additionalWorkflowProperties}};\n      if (this.config.modeSettings.sendUserPrompt) {\n        this.openChat([systemMsg, userMsg, userQueryMsg]);\n        this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));\n        this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(userMsg, 1));\n        this.chatService.generateAuditEvent('message', {...this._defineMessageAuditDetails(userQueryMsg, 2), 'query': JSON.stringify(this.query) ,'is-user-input': true, 'forced-workflow': this.config.modeSettings.initialization.forcedWorkflow, 'enabled-functions': this.config.defaultValues.functions?.filter(func => func.enabled).map(func => func.name), 'additional-workflow-properties':JSON.stringify(this.config.additionalWorkflowProperties)});\n      } else {\n        this.openChat([systemMsg, userQueryMsg]);\n        this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(systemMsg, 0));\n        this.chatService.generateAuditEvent('message', {...this._defineMessageAuditDetails(userQueryMsg, 1), 'query': JSON.stringify(this.query) ,'is-user-input': true, 'forced-workflow': this.config.modeSettings.initialization.forcedWorkflow, 'enabled-functions': this.config.defaultValues.functions?.filter(func => func.enabled).map(func => func.name), 'additional-workflow-properties': JSON.stringify(this.config.additionalWorkflowProperties)});\n      }\n    } else {\n      const warningMsg = {role: 'search-warning', content: this.config.globalSettings.searchWarningMessage, additionalProperties: {display: true}};\n      this.openChat([systemMsg, warningMsg]);\n      this.chatService.generateAuditEvent('message', this._defineMessageAuditDetails(warningMsg, 0));\n    }\n  }\n\n  private _defineMessageAuditDetails(message: ChatMessage, rank: number): Record<string, any> {\n    return {\n      'duration': 0,\n      'text': message.content,\n      'role': message.role,\n      'rank': rank\n    };\n  }\n\n  /**\n   * Start/open a new chat with the provided messages and chatId\n   * If the last message is from the user, a request to the assistant is made to get an answer\n   * If the last message is from the assistant, the conversation is loaded right away\n   * @param messages The list of messages of the chat\n   * @param savedChatId  The id of the saved chat. If provided (ie. an existing discussion in the saved chat index), update the savedChatId in the chat service for the upcoming saved chat operations\n   */\n  openChat(messages: RawMessage[], savedChatId?: string) {\n    if (!messages || !Array.isArray(messages)) {\n      console.error('Error occurs while trying to load the discussion. Invalid messages received :', messages);\n      return;\n    }\n    if (savedChatId) {\n      this.chatService.setSavedChatId(savedChatId);\n      this.chatService.generateChatId(savedChatId);\n    }\n    this.resetChat();\n    this.messages$.next(messages);\n    this.chatService.chatHistory = messages;\n    const lastMessage = messages.at(-1);\n    if(lastMessage && lastMessage.role === 'user') {\n      this.fetch(messages); // If the last message if from a user, an answer from the assistant is expected\n    }\n    else {\n      this.updateData(messages); // If the last message if from the assistant, we can load the conversation right away\n      this.terminateFetch();\n    }\n    this._addScrollListener();\n  }\n\n  /**\n   * Reset the chat by clearing the chat history and the UI accordingly\n   * The user input will be cleared\n   * The fetch subscription will be terminated\n   */\n  resetChat() {\n    if(this.messages$.value) {\n      this.messages$.next(undefined); // Reset chat\n    }\n    this.chatService.chatHistory = undefined; // Reset chat history\n    this.question = '';\n    this.terminateFetch();\n  }\n\n  /**\n   * Fetch and Load the saved chat from the saved chat index.\n   * If the saved chat is found, the chat discussion will be loaded with the provided messages and chatId\n   */\n  onLoadChat() {\n    this.loading$.next(true);\n    this._sub.add(\n      this.chatService.loadSavedChat$\n        .pipe(\n          filter(savedChat => !!savedChat),\n          switchMap(savedChat => this.chatService.getSavedChat(savedChat!.id)),\n          filter(savedChatHistory => !!savedChatHistory),\n          tap(savedChatHistory => this.openChat(savedChatHistory!.history, savedChatHistory!.id))\n        ).subscribe()\n    );\n  }\n\n  /**\n   * Stop the generation of the current assistant's answer.\n   * The fetch subscription will be terminated.\n   */\n  stopGeneration() {\n    this.chatService.stopGeneration().subscribe(\n      () => this.terminateFetch()\n    );\n  }\n\n  /**\n   * Terminate the fetch process by unsubscribing from the data subscription and updating the loading status to false.\n   * Additionally, focus on the chat input if the focusAfterResponse flag is set to true.\n   */\n  terminateFetch() {\n    this._dataSubscription?.unsubscribe();\n    this._dataSubscription = undefined;\n    this.loading$.next(false);\n    this.cdr.detectChanges();\n\n    if (this.focusAfterResponse) {\n      setTimeout(() => {\n        this.questionInput?.nativeElement.focus();\n      });\n    }\n  }\n\n  /**\n   * Copy a previous user message of the chat history to the chat user input.\n   * Thus, the user can edit and resubmit the message.\n   * Once the edited message is submitted, all subsequent messages starting from @param index will be removed from the history and the UI will be updated accordingly.\n   * The assistant will regenerate a new answer based on the updated chat history.\n   * ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.\n   * @param index The index of the user's message to edit\n   */\n  editMessage(index: number) {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.messageToEdit = index;\n    this.remappedMessageToEdit = this._remapIndexInChatHistory(index);\n    this.question = this.chatService.chatHistory![this._remapIndexInChatHistory(index)].content;\n    this.chatService.generateAuditEvent('edit.click', {'rank': this._remapIndexInChatHistory(index)});\n  }\n\n  /**\n   * Copy a previous assistant message of the chat history to the clipboard.\n   * @param index The index of the assistant's message to edit\n   */\n  copyMessage(index: number) {\n    // Remap the index in the chat history\n    const idx = this._remapIndexInChatHistory(index);\n    this.chatService.generateAuditEvent('copy.click', {'rank': idx});\n  }\n\n  /**\n   * Starting from the provided index, remove all subsequent messages from the chat history and the UI accordingly.\n   * The assistant will regenerate a new answer based on the updated chat history.\n   * ⚠️ If the assistant is streaming or stopping the generation, the operation is not allowed.\n   * @param index The index of the assistant's message to regenerate\n   */\n  regenerateMessage(index: number) {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    // Update the messages in the UI by removing all subsequent 'assistant' messages starting from the provided index until the first previous 'user' message\n    let i = index;\n    while (i >= 0 && (this.messages$.value!)[i].role !== 'user') {\n      i--;\n    }\n    // It should always be the case that i > 0\n    if (i >= 0) {\n      this.messages$.next(this.messages$.value!.slice(0, i+1));\n      // Remap the index of this found first previous 'user' message in the chat history\n      const idx = this._remapIndexInChatHistory(i);\n      // Define and Update the chat history based on which the assistant will generate a new answer\n      this.chatService.chatHistory = this.chatService.chatHistory!.slice(0, idx+1);\n      // Fetch the answer\n      this.fetch(this.chatService.chatHistory);\n      this.chatService.generateAuditEvent('regenerate.click', {'rank': idx});\n    }\n  }\n\n  /**\n   * Remaps the index in the chat history.\n   * The chat history is a list of messages where some messages can be hidden (display set to false).\n   * The index provided as input is the index of the message in the chat history displayed in the UI.\n   * This function should be removed once the backend is updated to add the ids of the messages in the chat history\n   * @param index - The index to be remapped.\n   */\n  private _remapIndexInChatHistory(index: number) {\n    // a copy of the chat history is created to avoid modifying the original chat history.\n    // Additionally, a rank is giving to each message.\n    // All messages having role 'user' are updated with the display property set to true, this is mandatory to get the correct rank of the message in the chat history when the user message to remap is hidden\n    const history = this.chatService.chatHistory!\n                      .slice()\n                      .map((message, idx) => ({...message, additionalProperties: {...message.additionalProperties, rank: idx}}))\n                      .map((message) => {\n                        if (message.role === \"user\") {\n                          return {...message, additionalProperties: {...message.additionalProperties, display: true}}\n                        }\n                        return message;\n                      })\n    // Count the number of hidden messages (of role different then \"user\") in messages$ before the provided index\n    // All messages having role 'user' are updated with the display property set to true, this is mandatory to get the correct rank of the message in the chat history when the user message to remap is hidden\n    // This is mandatory to get the correct rank of the message in the chat history\n    // Since some hidden messages (like 'system' messages) are not displayed in the UI but have been counted in the provided index\n    const numberOfHiddenMessagesInMessages$BeforeIndex = this.messages$.value!\n                                                            .slice(0, index)\n                                                            .map((message) => {\n                                                              if (message.role === \"user\") {\n                                                                return {...message, additionalProperties: {...message.additionalProperties, display: true}}\n                                                              }\n                                                              return message;\n                                                            })\n                                                            .filter(message => !message.additionalProperties.display).length;\n    // remove all messages that have display set to false\n    // this is mandatory since at the point of time when the assistant answers a question,\n    // it might have some hidden messages (for example contextMessages) that are available in the chat history but don't figure in messages$ unless a new question is asked\n    const filteredHistory = history.filter(message => message.additionalProperties.display);\n    // return the index of the message in the filtered history\n    return filteredHistory[index - numberOfHiddenMessagesInMessages$BeforeIndex].additionalProperties.rank;\n  }\n\n  /**\n   * Handles the key up event for 'Backspace' and 'Enter' keys.\n   * @param event - The keyboard event.\n   */\n  onKeyUp(event: KeyboardEvent): void {\n    switch (event.key) {\n      case 'Backspace':\n        this.calculateHeight();\n        break;\n      case 'Enter':\n        if (!event.shiftKey) {\n          event.preventDefault();\n          this.submitQuestion();\n        }\n        this.calculateHeight();\n        break;\n      default:\n        break;\n    }\n  }\n\n  /**\n   * Calculates and adjusts the height of the question input element based on its content.\n   * If the Enter key is pressed without the Shift key, it prevents the default behavior.\n   * @param event The keyboard event\n   */\n  calculateHeight(event?: KeyboardEvent): void {\n    if (event?.key === 'Enter' && !event.shiftKey) {\n      event?.preventDefault();\n    }\n    const maxHeight = 170;\n    const el = this.questionInput!.nativeElement;\n    el.style.maxHeight = `${maxHeight}px`;\n    el.style.height = 'auto';\n    el.style.height = `${el.scrollHeight}px`;\n    el.style.overflowY = el.scrollHeight >= maxHeight ? 'scroll' : 'hidden';\n  }\n\n  /**\n   * Send a \"like\" event on clicking on the thumb-up icon of an assistant's message\n   * @param message The assistant message to like\n   * @param rank The rank of the message to like\n   */\n  onLike(message: ChatMessage, rank: number): void {\n    // Remap the index in the chat history\n    const idx = this._remapIndexInChatHistory(rank);\n    this.chatService.generateAuditEvent('thumb-up.click', {rank: idx});\n    this.reportType = 'like';\n    this.messageToReport = message;\n    this.reportComment = undefined;\n    this.reportRank = rank;\n    this.showReport = true\n\n    this.chatService.chatHistory![this._remapIndexInChatHistory(rank)].additionalProperties.$liked = true;\n    this._updateChatHistory();\n  }\n\n  /**\n   * Send a \"dislike\" event on clicking on the thumb-down icon of an assistant's message.\n   * It also opens the issue reporting dialog.\n   * @param message The assistant message to dislike\n   * @param index The rank of the message to dislike\n   */\n  onDislike(message: ChatMessage, rank: number): void {\n    // Remap the index in the chat history\n    const idx = this._remapIndexInChatHistory(rank);\n    this.chatService.generateAuditEvent('thumb-down.click', {rank: idx});\n    this.reportType = 'dislike';\n    this.messageToReport = message;\n    this.issueType = '';\n    this.reportComment = undefined;\n    this.reportRank = rank;\n    this.showReport = true\n\n    this.chatService.chatHistory![this._remapIndexInChatHistory(rank)].additionalProperties.$disliked = true;\n    this._updateChatHistory();\n  }\n\n  private _updateChatHistory(): void {\n    this.messages$.next(this.chatService.chatHistory);\n    if (this.chatService.savedChatId) {\n      this.chatService.updateSavedChat(this.chatService.savedChatId, undefined, this.chatService.chatHistory).subscribe();\n    }\n  }\n\n  /**\n   * Report an issue related to the assistant's message.\n   */\n  sendReport(): void {\n    const details = {\n      'comment': this.reportComment,\n      'text': this.messageToReport!.content,\n      'rank': this.reportRank,\n    };\n    if (this.reportType === 'dislike') {\n      details['report-type'] = this.issueType;\n      this.chatService.generateAuditEvent('negative-report.send', details);\n    } else {\n      this.chatService.generateAuditEvent('positive-report.send', details);\n    }\n    this.notificationsService.success('Your report has been successfully sent');\n    this.showReport = false;\n  }\n\n  /**\n   * Close the reporting dialog.\n   */\n  ignoreReport(): void {\n    this.showReport = false;\n  }\n\n  /**\n   * Handle the click on a reference's 'open preview'.\n   * @param data\n   */\n  openAttachmentPreview(data: {reference: ChatContextAttachment, partId?: number}) {\n    this.openPreview.emit(data.reference);\n    const details = {\n      'doc-id': data.reference.recordId,\n      'title': data.reference.record.title,\n      'source': data.reference.record.treepath,\n      'collection': data.reference.record.collection,\n      'index': data.reference.record.databasealias,\n    };\n    if(!!data.partId) details['part-id'] = data.partId;\n    this.chatService.generateAuditEvent('attachment.preview.click', details);\n  }\n\n  /**\n   * Handle the click on a reference's 'open original document'.\n   * @param data\n   */\n  openOriginalAttachment(data: {reference: ChatContextAttachment, partId?: number}) {\n    this.openDocument.emit(data.reference.record);\n    const details = {\n      'doc-id': data.reference.recordId,\n      'title': data.reference.record.title,\n      'source': data.reference.record.treepath,\n      'collection': data.reference.record.collection,\n      'index': data.reference.record.databasealias,\n    };\n    if(!!data.partId) details['part-id'] = data.partId;\n    this.chatService.generateAuditEvent('attachment.link.click', details);\n  }\n\n  /**\n   * Handle the click on a suggested action.\n   * @param action Suggested action.\n   * @param index Rank of the message in the chatHistory related to the suggested action.\n   */\n  suggestActionClick(action: SuggestedAction, index: number) {\n    this.suggestAction.emit(action);\n    this.chatService.generateAuditEvent('suggestedAction.click', {'text': action.content, 'suggestedAction-type': action.type})\n  }\n\n  /**\n   * It looks for the debug messages available in the current group of \"assistant\" messages.\n   * By design, the debug messages are only available in the first visible message among the group \"assistant\" messages.\n   * @param index The rank of the message\n   * @returns The debug messages available in the current group of \"assistant\" messages\n   */\n  getDebugMessages(index: number): DebugMessage[] {\n    // If it is not an assistant message, return\n    if ((this.messages$.value!)[index].role !== 'assistant') {\n      return [];\n    }\n    // Get the array of messages up to the indicated index\n    const array = this.messages$.value!.slice(0, index + 1);\n    // If it is an assistant message, look for the debug messages available in the current group of \"assistant\" messages\n    // By design, the debug messages are only available in the first visible message among the group \"assistant\" messages.\n    const idx = this.chatService.firstVisibleAssistantMessageIndex(array);\n    if (idx > -1) {\n      return (this.messages$.value!)[idx].additionalProperties.$debug || [];\n    }\n    return [];\n  }\n\n  /**\n   * Handle the click on the 'show log info' button of a message.\n   * @param index The rank of the message\n   */\n  showDebug(index: number): void {\n    this.debugMessages = this.getDebugMessages(index);\n    this.showDebugMessages = true;\n    this.cdr.detectChanges();\n  }\n\n  /**\n   * Verify whether the current message is an assistant message and that all following messages are assistant ones\n   * Used to keep the \"View progress\" opened even though the assistant is sending additional messages after the current one\n   * @param messages the list of current messages\n   * @param index the index of the current message\n   * @returns if this messages and the following ones (if any) are the last ones\n   */\n  isAssistantLastMessages(messages: ChatMessage[], index: number): boolean {\n    for (let i = index; i < messages.length; i++) {\n      if (messages[i].role !== 'assistant') return false;\n    }\n    return true;\n  }\n\n  /**\n   * Checks if the given message is an empty assistant message.\n   * An empty assistant message is defined as a message with the role 'assistant',\n   * an empty content, and no additional properties such as attachments, progress,\n   * debug information, or suggested actions.\n   *\n   * @param message - The message to check.\n   * @returns `true` if the message is an empty assistant message, `false` otherwise.\n   */\n  isEmptyAssistantMessage(message: ChatMessage | undefined): boolean {\n    if (message?.role === 'assistant'\n          && message?.content === \"\"\n          && !message?.additionalProperties?.$attachment\n          && !message?.additionalProperties?.$progress\n          && !message?.additionalProperties?.$debug\n          && !message?.additionalProperties?.$suggestedAction) {\n      return true;\n    }\n    return false;\n  }\n}\n","<ng-container *ngIf=\"!initializationError\">\n  <div *ngIf=\"messages$ | async as messages; else loadingTpl || loadingTplDefault\" class=\"h-100 d-flex flex-column\">\n    <!-- Token consumption -->\n    <div class=\"ms-1\" *ngIf=\"config?.globalSettings?.displayUserQuotaConsumption || config?.globalSettings?.displayChatTokensConsumption\">\n      <ng-container *ngTemplateOutlet=\"tokenConsumptionTpl || defaultTokenConsumptionTpl; context: { $implicit: instanceId }\"></ng-container>\n    </div>\n\n    <!-- Chat Messages -->\n    <ul class=\"list-group list-group-flush overflow-auto flex-grow-1 pe-2 pb-2\" #messageList>\n      <ng-container *ngFor=\"let message of messages; let index = index; let last = last\">\n        <!-- Regular messages -->\n        <li class=\"list-group-item\"\n          *ngIf=\"message.additionalProperties.display && !isEmptyAssistantMessage(message)\"\n          [style.--bs-list-group-item-padding-y.rem]=\"'0.6'\"\n          [class.opacity-50]=\"messageToEdit && (messageToEdit < (index + 1))\">\n          <sq-chat-message\n            [class.sq-user-message]=\"message.role === 'user'\"\n            [class.last-message]=\"last\"\n            [message]=\"message\"\n            [conversation]=\"messages\"\n            [suggestedActions]=\"last ? message.additionalProperties.$suggestedAction : undefined\"\n            [assistantMessageIcon]=\"assistantMessageIcon\"\n            [userMessageIcon]=\"userMessageIcon\"\n            [connectionErrorMessageIcon]=\"connectionErrorMessageIcon\"\n            [searchWarningMessageIcon]=\"searchWarningMessageIcon\"\n            [streaming]=\"(chatService.streaming$ | async) && (last || isAssistantLastMessages(messages, index))\"\n            [canEdit]=\"(chatService.streaming$ | async) === false && messageToEdit === undefined && message.role === 'user'\"\n            [canCopy]=\"((chatService.streaming$ | async) === false || !last) && messageToEdit === undefined && message.role !== 'connection-error' && message.role !== 'search-warning'\"\n            [canLike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n            [canDislike]=\"((chatService.streaming$ | async) === false || !last) && message.role === 'assistant'\"\n            [canDebug]=\"(((chatService.streaming$ | async) === false && last) || (!last && messages[index+1].role !== 'assistant')) && message.role === 'assistant' && isAdmin && (getDebugMessages(index).length > 0) && config?.defaultValues.debug\"\n            [canRegenerate]=\"(chatService.streaming$ | async) === false  && (last || (!last && messages[index+1].role !== 'assistant'))  && message.role === 'assistant' && messageToEdit === undefined\"\n            (edit)=\"editMessage(index)\"\n            (copy)=\"copyMessage(index)\"\n            (regenerate)=\"regenerateMessage(index)\"\n            (openDocument)=\"openOriginalAttachment($event)\"\n            (openPreview)=\"openAttachmentPreview($event)\"\n            (suggestAction)=\"suggestActionClick($event, index)\"\n            (like)=\"onLike(message, index)\"\n            (dislike)=\"onDislike(message, index)\"\n            (debug)=\"showDebug(index)\">\n          </sq-chat-message>\n        </li>\n      </ng-container>\n      <!-- Loading spinner -->\n      <li class=\"list-group-item\" *ngIf=\"(loading$ | async) === true\">\n        <ng-container *ngTemplateOutlet=\"loadingTpl || loadingTplDefault\"></ng-container>\n      </li>\n    </ul>\n\n    <!-- Reporting a feedback form -->\n    <div class=\"issue-report pt-3 pb-2\" *ngIf=\"showReport\">\n      <ng-container *ngTemplateOutlet=\"reportTpl || reportTplDefault; context: { $implicit: messageToReport, rank: reportRank, type: reportType }\"></ng-container>\n    </div>\n\n    <!-- User text input -->\n    <div class=\"user-input mt-auto\" *ngIf=\"!showReport\">\n      <div class=\"py-2\">\n        <div [hidden]=\"!isConnected\">\n          <ng-container *ngIf=\"enabledUserInput\" [ngTemplateOutlet]=\"inputTpl\"></ng-container>\n        </div>\n        <!-- Retry button -->\n        <button [hidden]=\"isConnected\" class=\"btn mb-4 ast-error ast-btn sq-retry\" (click)=\"retryFetch()\">\n          <span>Try again</span>\n          <span *ngIf=\"retrialAttempts\" class=\"ms-2 attempts\">{{ retrialAttempts }}</span>\n        </button>\n        <div class=\"text-end small text-muted px-3\" *ngIf=\"!!config?.globalSettings?.disclaimer\">\n          {{ config?.globalSettings?.disclaimer }}\n        </div>\n      </div>\n    </div>\n\n    <!-- Floating scroll button -->\n    <div *ngIf=\"!isAtBottom && !showReport\" class=\"sq-floating-scroll\" [ngClass]=\"enabledUserInput ? 'sq-floating-scroll--when-user-input' : 'sq-floating-scroll--without-user-input'\">\n      <button class=\"btn shadow\" (click)=\"scrollDown()\">\n        <i class=\"fas fa-angle-double-down\"></i>\n      </button>\n    </div>\n  </div>\n</ng-container>\n\n<!-- NG TEMPLATES-->\n\n<ng-template #loadingTplDefault>\n  <div class=\"spinner-grow text-primary d-block mx-auto my-5\" role=\"status\">\n    <span class=\"visually-hidden\">Loading...</span>\n  </div>\n</ng-template>\n\n<ng-template #inputTpl>\n  <div class=\"px-3 py-1\">\n    <div class=\"ast-input-container\">\n      <button disabled class=\"btn btn-light\">\n        <i class=\"fas fa-search\"></i>\n      </button>\n      <textarea #questionInput rows=\"1\"\n        type=\"text\" class=\"form-control\"\n        placeholder=\"Ask something\" autofocus\n        [(ngModel)]=\"question\"\n        (keyup)=\"onKeyUp($event)\"\n        (keydown)=\"calculateHeight($event)\"\n        [disabled]=\"(loading$ | async) || (chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\">\n      </textarea>\n      <button\n        *ngIf=\"!(chatService.streaming$ | async) && !(loading$ | async) && !(chatService.stoppingGeneration$ | async)\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        sqTooltip=\"Send message\"\n        (click)=\"submitQuestion()\">\n        <i class=\"fas fa-paper-plane\"></i>\n      </button>\n      <button\n        *ngIf=\"messageToEdit\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        sqTooltip=\"Cancel edition\"\n        (click)=\"messageToEdit = undefined; question = ''\">\n        <i class=\"fas fa-undo-alt\"></i>\n      </button>\n      <span *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n        Generating <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n      </span>\n      <span *ngIf=\"(chatService.stoppingGeneration$ | async)\" class=\"processing ms-2\">\n        Stopping <i class=\"ms-1 fas fa-spinner fa-pulse\"></i>\n      </span>\n      <button\n        *ngIf=\"(chatService.streaming$ | async) && !(chatService.stoppingGeneration$ | async)\"\n        type=\"button\"\n        class=\"btn btn-light ms-2\"\n        sqTooltip=\"Stop generating\"\n        (click)=\"stopGeneration()\">\n        <i class=\"fas fa-stop\"></i>\n      </button>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #reportTplDefault let-message let-rank=\"rank\" let-type=\"type\">\n  <div class=\"px-3\">\n    <ng-container *ngIf=\"type === 'dislike'\">\n      <h5>Issue type</h5>\n      <select class=\"form-select mb-4\" [(ngModel)]=\"issueType\">\n        <option [value]=\"''\">Choose an issue type</option>\n        <option *ngFor=\"let type of (issueTypes ?? defaultIssueTypes)\">{{type}}</option>\n      </select>\n      <h5>What was unsatisfying about this response? (optional)</h5>\n    </ng-container>\n    <ng-container *ngIf=\"type === 'like'\">\n      <h5>Why did you like this answer? (optional)</h5>\n    </ng-container>\n    <textarea class=\"form-control\" [(ngModel)]=\"reportComment\" placeholder=\"Write your comment\"></textarea>\n    <div class=\"d-flex flex-row-reverse gap-1 mt-2\">\n      <button class=\"btn btn-primary\" [disabled]=\"type === 'dislike' && !issueType\" (click)=\"sendReport()\">Send</button>\n      <button class=\"btn btn-light\" (click)=\"ignoreReport()\">Do not send report</button>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #defaultTokenConsumptionTpl let-instanceId>\n  <sq-token-progress-bar\n    [instanceId]=\"instanceId\">\n  </sq-token-progress-bar>\n</ng-template>\n\n<div class=\"debug-messages\" [class.displayed]=\"showDebugMessages\">\n  <button *ngIf=\"showDebugMessages\" class=\"btn btn-light shadow back-btn\" (click)=\"showDebugMessages=false\">\n    <i class=\"fas fa-chevron-right\"></i>\n  </button>\n  <ng-container *ngTemplateOutlet=\"debugMessagesTpl || defaultDebugMessagesTpl; context: { $implicit: debugMessages }\">\n  </ng-container>\n</div>\n\n<ng-template #defaultDebugMessagesTpl let-debugMessages>\n  <sq-debug-message [data]=\"debugMessages\"></sq-debug-message>\n</ng-template>\n"]}
|