@sinequa/assistant 3.9.2 → 3.9.3
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.component.d.ts +35 -38
- package/chat/chat.service.d.ts +190 -84
- package/chat/debug-message/debug-message-details/debug-message-details.component.d.ts +19 -0
- package/chat/debug-message/debug-message.component.d.ts +9 -12
- package/chat/debug-message/debug-message.service.d.ts +15 -0
- package/chat/documents-upload/document-list/document-list.component.d.ts +1 -0
- package/chat/public-api.d.ts +0 -2
- package/chat/saved-chats/i18n/en.json +1 -1
- package/chat/saved-chats/i18n/fr.json +1 -1
- package/chat/saved-chats/saved-chats.component.d.ts +1 -0
- package/chat/saved-chats/saved-chats.service.d.ts +27 -0
- package/chat/services/assistant-configuration.service.d.ts +34 -0
- package/chat/services/assistant-metadata.service.d.ts +18 -0
- package/chat/services/assistant-tokens-tracking.service.d.ts +23 -0
- package/chat/services/assistant-ws-frames.service.d.ts +50 -0
- package/chat/services/signalR-connection.service.d.ts +25 -0
- package/chat/types.d.ts +12 -3
- package/chat/utils/utils.service.d.ts +67 -0
- package/esm2022/chat/chat.component.mjs +198 -227
- package/esm2022/chat/chat.service.mjs +450 -267
- package/esm2022/chat/debug-message/debug-message-details/debug-message-details.component.mjs +43 -0
- package/esm2022/chat/debug-message/debug-message.component.mjs +27 -30
- package/esm2022/chat/debug-message/debug-message.service.mjs +52 -0
- package/esm2022/chat/dialogs/rename-saved-chat.component.mjs +6 -5
- package/esm2022/chat/documents-upload/document-list/document-list.component.mjs +4 -2
- package/esm2022/chat/public-api.mjs +1 -3
- package/esm2022/chat/saved-chats/saved-chats.component.mjs +11 -10
- package/esm2022/chat/saved-chats/saved-chats.service.mjs +165 -0
- package/esm2022/chat/services/assistant-configuration.service.mjs +171 -0
- package/esm2022/chat/services/assistant-metadata.service.mjs +67 -0
- package/esm2022/chat/services/assistant-tokens-tracking.service.mjs +57 -0
- package/esm2022/chat/services/assistant-ws-frames.service.mjs +392 -0
- package/esm2022/chat/services/signalR-connection.service.mjs +109 -0
- package/esm2022/chat/types.mjs +1 -1
- package/esm2022/chat/unified-plugins/embedded-image-reference.plugin.mjs +2 -3
- package/esm2022/chat/utils/utils.service.mjs +170 -0
- package/fesm2022/sinequa-assistant-chat.mjs +2020 -1648
- package/fesm2022/sinequa-assistant-chat.mjs.map +1 -1
- package/package.json +1 -1
- package/chat/rest-chat.service.d.ts +0 -31
- package/chat/websocket-chat.service.d.ts +0 -102
- package/esm2022/chat/rest-chat.service.mjs +0 -300
- package/esm2022/chat/websocket-chat.service.mjs +0 -659
|
@@ -5,6 +5,7 @@ import { BehaviorSubject, Subscription, combineLatest, filter, of, switchMap, ta
|
|
|
5
5
|
import { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';
|
|
6
6
|
import { parseISO, toDate } from "date-fns";
|
|
7
7
|
import { isAuthenticated } from "@sinequa/atomic";
|
|
8
|
+
import { AssistantUtils } from "../utils/utils.service";
|
|
8
9
|
import { DialogDeleteSavedChatComponent } from "../dialogs/delete-saved-chat.component";
|
|
9
10
|
import { DialogRenameSavedChatComponent } from "../dialogs/rename-saved-chat.component";
|
|
10
11
|
import { InstanceManagerService } from "../instance-manager.service";
|
|
@@ -21,11 +22,14 @@ export class SavedChatsComponent {
|
|
|
21
22
|
this.instanceManagerService = inject(InstanceManagerService);
|
|
22
23
|
this.notificationsService = inject(NotificationsService);
|
|
23
24
|
this.transloco = inject(TranslocoService);
|
|
25
|
+
this.assistantUtils = inject(AssistantUtils);
|
|
24
26
|
}
|
|
25
27
|
ngOnInit() {
|
|
26
28
|
this.subscription.add(of(isAuthenticated()).pipe(tap(_ => this.instantiateChatService()), switchMap(_ => this.chatService.userOverride$), switchMap(_ => this.chatService.initProcess$), filter(success => !!success), tap(_ => {
|
|
27
29
|
this.onListSavedChat();
|
|
28
|
-
this.chatService.
|
|
30
|
+
if (!!this.chatService.assistantConfig$.value.savedChatSettings?.enabled) {
|
|
31
|
+
this.chatService.listSavedChat();
|
|
32
|
+
}
|
|
29
33
|
})).subscribe());
|
|
30
34
|
}
|
|
31
35
|
ngOnDestroy() {
|
|
@@ -38,18 +42,15 @@ export class SavedChatsComponent {
|
|
|
38
42
|
this.subscription.add(combineLatest([
|
|
39
43
|
this.chatService.savedChats$,
|
|
40
44
|
this.transloco.langChanges$
|
|
41
|
-
]).subscribe(([savedChats]) =>
|
|
42
|
-
this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats));
|
|
43
|
-
}));
|
|
45
|
+
]).pipe(filter(([savedChats]) => !!savedChats)).subscribe(([savedChats]) => this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats))));
|
|
44
46
|
}
|
|
45
47
|
onLoad(savedChat) {
|
|
46
48
|
if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {
|
|
47
49
|
return;
|
|
48
50
|
}
|
|
49
|
-
this.chatService.setSavedChatId(savedChat.id);
|
|
50
51
|
this.chatService.generateChatId(savedChat.id);
|
|
51
52
|
this.chatService.loadSavedChat$.next(savedChat);
|
|
52
|
-
this.chatService.listSavedChat();
|
|
53
|
+
this.chatService.listSavedChat(); // assuming saved chats operations are certainly enabled at this level
|
|
53
54
|
this.load.emit(savedChat);
|
|
54
55
|
}
|
|
55
56
|
_groupSavedChatsByDate(savedChats) {
|
|
@@ -57,7 +58,7 @@ export class SavedChatsComponent {
|
|
|
57
58
|
savedChats
|
|
58
59
|
.sort((a, b) => parseISO(b.modifiedUTC).getTime() - parseISO(a.modifiedUTC).getTime())
|
|
59
60
|
.forEach(savedChat => {
|
|
60
|
-
const groupKey = this.
|
|
61
|
+
const groupKey = this.assistantUtils.getTimeKey(toDate(parseISO(savedChat.modifiedUTC))); // Must convert the UTC date to local date before passing to _getTimeKey
|
|
61
62
|
if (!groupedSavedChats.has(groupKey)) {
|
|
62
63
|
groupedSavedChats.set(groupKey, []);
|
|
63
64
|
}
|
|
@@ -67,11 +68,11 @@ export class SavedChatsComponent {
|
|
|
67
68
|
;
|
|
68
69
|
}
|
|
69
70
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SavedChatsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
70
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: SavedChatsComponent, isStandalone: true, selector: "sq-saved-chats-v3", inputs: { instanceId: "instanceId" }, outputs: { load: "load", delete: "delete" }, providers: [provideTranslocoScope('saved-chats', 'chat')], ngImport: i0, template: "<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n <div *ngIf=\"chatService.savedChatsError$ | async\" class=\"alert alert-danger\">\n {{ 'savedChats.
|
|
71
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: SavedChatsComponent, isStandalone: true, selector: "sq-saved-chats-v3", inputs: { instanceId: "instanceId" }, outputs: { load: "load", delete: "delete" }, providers: [provideTranslocoScope('saved-chats', 'chat')], ngImport: i0, template: "<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n <div *ngIf=\"chatService.savedChatsError$ | async\" class=\"alert alert-danger\">\n {{ 'savedChats.listError' | transloco }}\n </div>\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\"\n (click)=\"onLoad(savedChat)\"\n class=\"saved-chat p-2\"\n [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n [class.active]=\"chatService.chatId === savedChat.id\">\n <span class=\"title me-1\" [sqTooltip]=\"savedChat.title\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'savedChats.rename' | transloco\" (click)=\"renameDialog.showModal($event, savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'savedChats.delete' | transloco\" (click)=\"deleteDialog.showModal($event, savedChat)\"></i>\n </div>\n </div>\n</ng-container>\n\n<sq-dialog-delete-saved-chat [chatService]=\"chatService\" (delete)=\"delete.emit($event)\" #deleteDialog />\n<sq-dialog-rename-saved-chat [chatService]=\"chatService\" #renameDialog />\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-uploaded-doc-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}.saved-chats .saved-chat-date{font-size:12px;font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .saved-chat.forbidden{cursor:not-allowed}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }, { kind: "component", type: DialogRenameSavedChatComponent, selector: "sq-dialog-rename-saved-chat", inputs: ["chatService"] }, { kind: "component", type: DialogDeleteSavedChatComponent, selector: "sq-dialog-delete-saved-chat", inputs: ["chatService"], outputs: ["delete"] }, { kind: "pipe", type: TranslocoPipe, name: "transloco" }] }); }
|
|
71
72
|
}
|
|
72
73
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SavedChatsComponent, decorators: [{
|
|
73
74
|
type: Component,
|
|
74
|
-
args: [{ selector: 'sq-saved-chats-v3', standalone: true, providers: [provideTranslocoScope('saved-chats', 'chat')], imports: [CommonModule, FormsModule, TooltipDirective, DialogRenameSavedChatComponent, DialogDeleteSavedChatComponent, TranslocoPipe], template: "<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n <div *ngIf=\"chatService.savedChatsError$ | async\" class=\"alert alert-danger\">\n {{ 'savedChats.
|
|
75
|
+
args: [{ selector: 'sq-saved-chats-v3', standalone: true, providers: [provideTranslocoScope('saved-chats', 'chat')], imports: [CommonModule, FormsModule, TooltipDirective, DialogRenameSavedChatComponent, DialogDeleteSavedChatComponent, TranslocoPipe], template: "<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n <div *ngIf=\"chatService.savedChatsError$ | async\" class=\"alert alert-danger\">\n {{ 'savedChats.listError' | transloco }}\n </div>\n <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n <div class=\"saved-chat-date\">{{group.key}}</div>\n <div *ngFor=\"let savedChat of group.value\"\n (click)=\"onLoad(savedChat)\"\n class=\"saved-chat p-2\"\n [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n [class.active]=\"chatService.chatId === savedChat.id\">\n <span class=\"title me-1\" [sqTooltip]=\"savedChat.title\">{{savedChat.title}}</span>\n <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'savedChats.rename' | transloco\" (click)=\"renameDialog.showModal($event, savedChat)\"></i>\n <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'savedChats.delete' | transloco\" (click)=\"deleteDialog.showModal($event, savedChat)\"></i>\n </div>\n </div>\n</ng-container>\n\n<sq-dialog-delete-saved-chat [chatService]=\"chatService\" (delete)=\"delete.emit($event)\" #deleteDialog />\n<sq-dialog-rename-saved-chat [chatService]=\"chatService\" #renameDialog />\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-uploaded-doc-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}.saved-chats .saved-chat-date{font-size:12px;font-weight:500;color:#a9a9a9;margin-top:.5rem}.saved-chats .saved-chat{display:flex;align-items:center;cursor:pointer;margin-left:.25rem}.saved-chats .saved-chat span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.saved-chats .saved-chat .saved-chat-actions{display:none}.saved-chats .saved-chat:hover,.saved-chats .saved-chat.active{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-saved-chat-hover-background, #FFF8F1)}.saved-chats .saved-chat:hover .saved-chat-actions{display:block}.saved-chats .saved-chat.forbidden{cursor:not-allowed}.saved-chats .title{display:block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\n"] }]
|
|
75
76
|
}], propDecorators: { instanceId: [{
|
|
76
77
|
type: Input
|
|
77
78
|
}], load: [{
|
|
@@ -79,4 +80,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
79
80
|
}], delete: [{
|
|
80
81
|
type: Output
|
|
81
82
|
}] } });
|
|
82
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"saved-chats.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/saved-chats/saved-chats.component.ts","../../../../../projects/assistant/chat/saved-chats/saved-chats.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from \"@angular/common\";\nimport { Component, EventEmitter, Input, OnDestroy, OnInit, Output, inject } from \"@angular/core\";\nimport { FormsModule } from '@angular/forms';\nimport { BehaviorSubject, Subscription, combineLatest, filter, of, switchMap, tap } from \"rxjs\";\nimport { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';\nimport { parseISO, toDate } from \"date-fns\";\n\nimport { isAuthenticated } from \"@sinequa/atomic\";\n\nimport { ChatService } from \"../chat.service\";\nimport { DialogDeleteSavedChatComponent } from \"../dialogs/delete-saved-chat.component\";\nimport { DialogRenameSavedChatComponent } from \"../dialogs/rename-saved-chat.component\";\nimport { InstanceManagerService } from \"../instance-manager.service\";\nimport { NotificationsService } from \"../services/notification.service\";\nimport { TooltipDirective } from \"../tooltip/tooltip.directive\";\nimport { SavedChat } from \"../types\";\n\n\n@Component({\n  selector: 'sq-saved-chats-v3',\n  templateUrl: 'saved-chats.component.html',\n  styleUrls: ['saved-chats.component.scss'],\n  standalone: true,\n  providers: [provideTranslocoScope('saved-chats', 'chat')],\n  imports: [CommonModule, FormsModule, TooltipDirective, DialogRenameSavedChatComponent, DialogDeleteSavedChatComponent, TranslocoPipe]\n})\nexport class SavedChatsComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n\n  @Output() load = new EventEmitter<SavedChat>();\n  @Output() delete = new EventEmitter<SavedChat>();\n\n  chatService: ChatService;\n  subscription = new Subscription();\n  groupedSavedChats$ = new BehaviorSubject<{ key: string, value: SavedChat[] }[]>([]);\n\n  public instanceManagerService = inject(InstanceManagerService);\n  public notificationsService = inject(NotificationsService);\n  private readonly transloco = inject(TranslocoService);\n\n  ngOnInit(): void {\n    this.subscription.add(\n      of(isAuthenticated()).pipe(\n        tap(_ => this.instantiateChatService()),\n        switchMap(_ => this.chatService.userOverride$),\n        switchMap(_ => this.chatService.initProcess$),\n        filter(success => !!success),\n        tap(_ => {\n          this.onListSavedChat();\n          this.chatService.listSavedChat();\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    this.chatService = this.instanceManagerService.getInstance(this.instanceId);\n  }\n\n  onListSavedChat() {\n    this.subscription.add(\n      combineLatest([\n        this.chatService.savedChats$,\n        this.transloco.langChanges$\n      ]).subscribe(([savedChats]) => {\n        this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats));\n      })\n    );\n  }\n\n  onLoad(savedChat: SavedChat) {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.chatService.setSavedChatId(savedChat.id);\n    this.chatService.generateChatId(savedChat.id);\n    this.chatService.loadSavedChat$.next(savedChat);\n    this.chatService.listSavedChat();\n    this.load.emit(savedChat);\n  }\n\n  private _groupSavedChatsByDate(savedChats: SavedChat[]): { key: string, value: SavedChat[] }[] {\n    const groupedSavedChats = new Map<string, SavedChat[]>();\n\n    savedChats\n      .sort((a, b) => parseISO(b.modifiedUTC).getTime() - parseISO(a.modifiedUTC).getTime())\n      .forEach(savedChat => {\n        const groupKey = this.chatService.getTimeKey(toDate(parseISO(savedChat.modifiedUTC))); // Must convert the UTC date to local date before passing to _getTimeKey\n\n        if (!groupedSavedChats.has(groupKey)) {\n          groupedSavedChats.set(groupKey, []);\n        }\n\n        groupedSavedChats.get(groupKey)!.push(savedChat);\n      });\n\n    return Array.from(groupedSavedChats, ([key, value]) => ({ key, value }));;\n  }\n}\n","<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n  <div *ngIf=\"chatService.savedChatsError$ | async\" class=\"alert alert-danger\">\n    {{ 'savedChats.error' | transloco }}\n  </div>\n  <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n    <div class=\"saved-chat-date\">{{group.key}}</div>\n    <div *ngFor=\"let savedChat of group.value\"\n      (click)=\"onLoad(savedChat)\"\n      class=\"saved-chat p-2\"\n      [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n      [class.active]=\"chatService.savedChatId === savedChat.id\">\n      <span class=\"title me-1\" [sqTooltip]=\"savedChat.title\">{{savedChat.title}}</span>\n      <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'savedChats.rename' | transloco\" (click)=\"renameDialog.showModal($event, savedChat)\"></i>\n      <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'savedChats.delete' | transloco\" (click)=\"deleteDialog.showModal($event, savedChat)\"></i>\n    </div>\n  </div>\n</ng-container>\n\n<sq-dialog-delete-saved-chat [chatService]=\"chatService\" (delete)=\"delete.emit($event)\" #deleteDialog />\n<sq-dialog-rename-saved-chat [chatService]=\"chatService\" #renameDialog />\n"]}
|
|
83
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"saved-chats.component.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/saved-chats/saved-chats.component.ts","../../../../../projects/assistant/chat/saved-chats/saved-chats.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from \"@angular/common\";\nimport { Component, EventEmitter, Input, OnDestroy, OnInit, Output, inject } from \"@angular/core\";\nimport { FormsModule } from '@angular/forms';\nimport { BehaviorSubject, Subscription, combineLatest, filter, of, switchMap, tap } from \"rxjs\";\nimport { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';\nimport { parseISO, toDate } from \"date-fns\";\n\nimport { isAuthenticated } from \"@sinequa/atomic\";\n\nimport { ChatService } from \"../chat.service\";\nimport { AssistantUtils } from \"../utils/utils.service\";\nimport { DialogDeleteSavedChatComponent } from \"../dialogs/delete-saved-chat.component\";\nimport { DialogRenameSavedChatComponent } from \"../dialogs/rename-saved-chat.component\";\nimport { InstanceManagerService } from \"../instance-manager.service\";\nimport { NotificationsService } from \"../services/notification.service\";\nimport { TooltipDirective } from \"../tooltip/tooltip.directive\";\nimport { SavedChat } from \"../types\";\n\n\n@Component({\n  selector: 'sq-saved-chats-v3',\n  templateUrl: 'saved-chats.component.html',\n  styleUrls: ['saved-chats.component.scss'],\n  standalone: true,\n  providers: [provideTranslocoScope('saved-chats', 'chat')],\n  imports: [CommonModule, FormsModule, TooltipDirective, DialogRenameSavedChatComponent, DialogDeleteSavedChatComponent, TranslocoPipe]\n})\nexport class SavedChatsComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n\n  @Output() load = new EventEmitter<SavedChat>();\n  @Output() delete = new EventEmitter<SavedChat>();\n\n  chatService: ChatService;\n  subscription = new Subscription();\n  groupedSavedChats$ = new BehaviorSubject<{ key: string, value: SavedChat[] }[]>([]);\n\n  public instanceManagerService = inject(InstanceManagerService);\n  public notificationsService = inject(NotificationsService);\n  private readonly transloco = inject(TranslocoService);\n  private readonly assistantUtils = inject(AssistantUtils);\n\n  ngOnInit(): void {\n    this.subscription.add(\n      of(isAuthenticated()).pipe(\n        tap(_ => this.instantiateChatService()),\n        switchMap(_ => this.chatService.userOverride$),\n        switchMap(_ => this.chatService.initProcess$),\n        filter(success => !!success),\n        tap(_ => {\n          this.onListSavedChat();\n          if (!!this.chatService.assistantConfig$.value!.savedChatSettings?.enabled) {\n            this.chatService.listSavedChat();\n          }\n        })\n      ).subscribe()\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n  }\n\n  instantiateChatService(): void {\n    this.chatService = this.instanceManagerService.getInstance(this.instanceId);\n  }\n\n  onListSavedChat() {\n    this.subscription.add(\n      combineLatest([\n        this.chatService.savedChats$,\n        this.transloco.langChanges$\n      ]).pipe(\n        filter(([savedChats]) => !!savedChats)\n      ).subscribe(([savedChats]) => this.groupedSavedChats$.next(this._groupSavedChatsByDate(savedChats)))\n    );\n  }\n\n  onLoad(savedChat: SavedChat) {\n    if (!!this.chatService.streaming$.value || !!this.chatService.stoppingGeneration$.value) {\n      return;\n    }\n    this.chatService.generateChatId(savedChat.id);\n    this.chatService.loadSavedChat$.next(savedChat);\n    this.chatService.listSavedChat(); // assuming saved chats operations are certainly enabled at this level\n    this.load.emit(savedChat);\n  }\n\n  private _groupSavedChatsByDate(savedChats: SavedChat[]): { key: string, value: SavedChat[] }[] {\n    const groupedSavedChats = new Map<string, SavedChat[]>();\n\n    savedChats\n      .sort((a, b) => parseISO(b.modifiedUTC).getTime() - parseISO(a.modifiedUTC).getTime())\n      .forEach(savedChat => {\n        const groupKey = this.assistantUtils.getTimeKey(toDate(parseISO(savedChat.modifiedUTC))); // Must convert the UTC date to local date before passing to _getTimeKey\n\n        if (!groupedSavedChats.has(groupKey)) {\n          groupedSavedChats.set(groupKey, []);\n        }\n\n        groupedSavedChats.get(groupKey)!.push(savedChat);\n      });\n\n    return Array.from(groupedSavedChats, ([key, value]) => ({ key, value }));;\n  }\n}\n","<ng-container *ngIf=\"(chatService.assistantConfig$ | async)?.savedChatSettings.display\">\n  <div *ngIf=\"chatService.savedChatsError$ | async\" class=\"alert alert-danger\">\n    {{ 'savedChats.listError' | transloco }}\n  </div>\n  <div *ngFor=\"let group of (groupedSavedChats$ | async)\" class=\"saved-chats\">\n    <div class=\"saved-chat-date\">{{group.key}}</div>\n    <div *ngFor=\"let savedChat of group.value\"\n      (click)=\"onLoad(savedChat)\"\n      class=\"saved-chat p-2\"\n      [class.forbidden]=\"(chatService.streaming$ | async) || (chatService.stoppingGeneration$ | async)\"\n      [class.active]=\"chatService.chatId === savedChat.id\">\n      <span class=\"title me-1\" [sqTooltip]=\"savedChat.title\">{{savedChat.title}}</span>\n      <i class=\"saved-chat-actions fas fa-pen mx-1\" [sqTooltip]=\"'savedChats.rename' | transloco\" (click)=\"renameDialog.showModal($event, savedChat)\"></i>\n      <i class=\"saved-chat-actions fas fa-trash ms-1\" [sqTooltip]=\"'savedChats.delete' | transloco\" (click)=\"deleteDialog.showModal($event, savedChat)\"></i>\n    </div>\n  </div>\n</ng-container>\n\n<sq-dialog-delete-saved-chat [chatService]=\"chatService\" (delete)=\"delete.emit($event)\" #deleteDialog />\n<sq-dialog-rename-saved-chat [chatService]=\"chatService\" #renameDialog />\n"]}
|
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { from, throwError } from 'rxjs';
|
|
3
|
+
import { tap, catchError, map } from 'rxjs/operators';
|
|
4
|
+
import { get, post } from '@sinequa/atomic';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
export class SavedChatsService {
|
|
7
|
+
constructor() { }
|
|
8
|
+
init(config) {
|
|
9
|
+
if (this.operationConfig) {
|
|
10
|
+
console.warn('SavedChatsService already initialized.');
|
|
11
|
+
return;
|
|
12
|
+
}
|
|
13
|
+
this.operationConfig = config;
|
|
14
|
+
}
|
|
15
|
+
ensureInitialized() {
|
|
16
|
+
if (!this.operationConfig) {
|
|
17
|
+
throw new Error('SavedChatsService not initialized. Call init() first.');
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
listChats() {
|
|
21
|
+
this.ensureInitialized();
|
|
22
|
+
const config = this.operationConfig;
|
|
23
|
+
if (!config.isSavedChatsEnabled()) {
|
|
24
|
+
throwError(() => new Error("Operations on saved chats are disabled"));
|
|
25
|
+
}
|
|
26
|
+
const data = {
|
|
27
|
+
action: "SavedChatList",
|
|
28
|
+
appName: config.getAppName(),
|
|
29
|
+
instanceId: config.getInstanceId(),
|
|
30
|
+
debug: config.getDebugFlag()
|
|
31
|
+
};
|
|
32
|
+
const searchParams = new URLSearchParams(Object.entries(data).reduce((acc, [key, value]) => {
|
|
33
|
+
acc[key] = String(value);
|
|
34
|
+
return acc;
|
|
35
|
+
}, {}));
|
|
36
|
+
from(get(`plugin/${config.getRestUrl()}`, searchParams)).pipe(tap(res => {
|
|
37
|
+
config.updateSavedChatsList(res.savedChats);
|
|
38
|
+
config.setSavedChatsErrorStatus(false);
|
|
39
|
+
}), catchError((error) => {
|
|
40
|
+
const errorMsg = 'Error occurred while calling the SavedChatList API: ' + error.error?.errorMessage;
|
|
41
|
+
console.error('SavedChatsService: ' + errorMsg);
|
|
42
|
+
config.setSavedChatsErrorStatus(true);
|
|
43
|
+
return throwError(() => error);
|
|
44
|
+
})).subscribe();
|
|
45
|
+
}
|
|
46
|
+
getChatById(id) {
|
|
47
|
+
this.ensureInitialized();
|
|
48
|
+
const config = this.operationConfig;
|
|
49
|
+
if (!config.isSavedChatsEnabled()) {
|
|
50
|
+
return throwError(() => new Error("Operations on saved chats are disabled"));
|
|
51
|
+
}
|
|
52
|
+
const data = {
|
|
53
|
+
action: "SavedChatGet",
|
|
54
|
+
appName: config.getAppName(),
|
|
55
|
+
instanceId: config.getInstanceId(),
|
|
56
|
+
chatId: id,
|
|
57
|
+
debug: config.getDebugFlag()
|
|
58
|
+
};
|
|
59
|
+
const searchParams = new URLSearchParams(Object.entries(data).reduce((acc, [key, value]) => {
|
|
60
|
+
acc[key] = String(value);
|
|
61
|
+
return acc;
|
|
62
|
+
}, {}));
|
|
63
|
+
return from(get(`plugin/${config.getRestUrl()}`, searchParams)).pipe(tap(res => {
|
|
64
|
+
config.generateAuditEvent('ast-saved-chat.load', { duration: res.executionTimeMilliseconds }, res.savedChat.id);
|
|
65
|
+
}), map(res => res.savedChat), catchError((error) => {
|
|
66
|
+
const errorMsg = 'Error occurred while calling the SavedChatGet API: ' + error.error?.errorMessage;
|
|
67
|
+
console.error('SavedChatsService: ' + errorMsg);
|
|
68
|
+
return throwError(() => error);
|
|
69
|
+
}));
|
|
70
|
+
}
|
|
71
|
+
addChat(id, messages) {
|
|
72
|
+
this.ensureInitialized();
|
|
73
|
+
const config = this.operationConfig;
|
|
74
|
+
if (!config.isSavedChatsEnabled()) {
|
|
75
|
+
return throwError(() => new Error("Operations on saved chats are disabled"));
|
|
76
|
+
}
|
|
77
|
+
const data = {
|
|
78
|
+
action: "SavedChatAdd",
|
|
79
|
+
appName: config.getAppName(),
|
|
80
|
+
instanceId: config.getInstanceId(),
|
|
81
|
+
chatId: id,
|
|
82
|
+
history: messages,
|
|
83
|
+
debug: config.getDebugFlag()
|
|
84
|
+
};
|
|
85
|
+
return from(post(`plugin/${config.getRestUrl()}`, data)).pipe(tap(res => {
|
|
86
|
+
config.generateAuditEvent('ast-saved-chat.add', { duration: res.executionTimeMilliseconds }, res.savedChat.id);
|
|
87
|
+
}), catchError((error) => {
|
|
88
|
+
const errorMsg = 'Error occurred while calling the SavedChatAdd API: ' + error.error?.errorMessage;
|
|
89
|
+
console.error('SavedChatsService: ' + errorMsg);
|
|
90
|
+
return throwError(() => error);
|
|
91
|
+
}));
|
|
92
|
+
}
|
|
93
|
+
updateChat(id, name, messages) {
|
|
94
|
+
this.ensureInitialized();
|
|
95
|
+
const config = this.operationConfig;
|
|
96
|
+
if (!config.isSavedChatsEnabled()) {
|
|
97
|
+
return throwError(() => new Error("Operations on saved chats are disabled"));
|
|
98
|
+
}
|
|
99
|
+
const data = {
|
|
100
|
+
action: "SavedChatUpdate",
|
|
101
|
+
appName: config.getAppName(),
|
|
102
|
+
instanceId: config.getInstanceId(),
|
|
103
|
+
chatId: id,
|
|
104
|
+
debug: config.getDebugFlag()
|
|
105
|
+
};
|
|
106
|
+
if (name)
|
|
107
|
+
data["title"] = name;
|
|
108
|
+
if (messages)
|
|
109
|
+
data["history"] = messages;
|
|
110
|
+
return from(post(`plugin/${config.getRestUrl()}`, data)).pipe(catchError((error) => {
|
|
111
|
+
const errorMsg = 'Error occurred while calling the SavedChatUpdate API: ' + error.error?.errorMessage;
|
|
112
|
+
console.error('SavedChatsService: ' + errorMsg);
|
|
113
|
+
return throwError(() => error);
|
|
114
|
+
}));
|
|
115
|
+
}
|
|
116
|
+
deleteChat(ids) {
|
|
117
|
+
this.ensureInitialized();
|
|
118
|
+
const config = this.operationConfig;
|
|
119
|
+
if (!config.isSavedChatsEnabled()) {
|
|
120
|
+
return throwError(() => new Error("Operations on saved chats are disabled"));
|
|
121
|
+
}
|
|
122
|
+
const data = {
|
|
123
|
+
action: "SavedChatDelete",
|
|
124
|
+
appName: config.getAppName(),
|
|
125
|
+
instanceId: config.getInstanceId(),
|
|
126
|
+
chatIds: ids,
|
|
127
|
+
debug: config.getDebugFlag()
|
|
128
|
+
};
|
|
129
|
+
return from(post(`plugin/${config.getRestUrl()}`, data)).pipe(catchError((error) => {
|
|
130
|
+
return throwError(() => error);
|
|
131
|
+
}));
|
|
132
|
+
}
|
|
133
|
+
isExistingChat(id) {
|
|
134
|
+
this.ensureInitialized();
|
|
135
|
+
const config = this.operationConfig;
|
|
136
|
+
if (!config.isSavedChatsEnabled()) {
|
|
137
|
+
return throwError(() => new Error("Operations on saved chats are disabled"));
|
|
138
|
+
}
|
|
139
|
+
const data = {
|
|
140
|
+
action: "SavedChatExist",
|
|
141
|
+
appName: config.getAppName(),
|
|
142
|
+
instanceId: config.getInstanceId(),
|
|
143
|
+
chatId: id,
|
|
144
|
+
debug: config.getDebugFlag()
|
|
145
|
+
};
|
|
146
|
+
const searchParams = new URLSearchParams(Object.entries(data).reduce((acc, [key, value]) => {
|
|
147
|
+
acc[key] = String(value);
|
|
148
|
+
return acc;
|
|
149
|
+
}, {}));
|
|
150
|
+
return from(get(`plugin/${config.getRestUrl()}`, searchParams)).pipe(map(res => res.exists), catchError((error) => {
|
|
151
|
+
const errorMsg = 'Error occurred while calling the SavedChatExist API: ' + error.error?.errorMessage;
|
|
152
|
+
console.error('SavedChatsService: ' + errorMsg);
|
|
153
|
+
return throwError(() => error);
|
|
154
|
+
}));
|
|
155
|
+
}
|
|
156
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SavedChatsService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
157
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SavedChatsService, providedIn: 'root' }); }
|
|
158
|
+
}
|
|
159
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SavedChatsService, decorators: [{
|
|
160
|
+
type: Injectable,
|
|
161
|
+
args: [{
|
|
162
|
+
providedIn: 'root'
|
|
163
|
+
}]
|
|
164
|
+
}], ctorParameters: () => [] });
|
|
165
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"saved-chats.service.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/saved-chats/saved-chats.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Observable, from, throwError } from 'rxjs';\nimport { tap, catchError, map } from 'rxjs/operators';\nimport { get, post } from '@sinequa/atomic';\n\nimport {\n  ChatMessage,\n  SavedChat,\n  SavedChatHistory,\n  SavedChatResponse,\n  DeleteSavedChatResponse\n} from '../types';\n\nexport interface SavedChatsOperationConfig {\n  getAppName: () => string;\n  getInstanceId: () => string;\n  getRestUrl: () => string;\n  getDebugFlag: () => boolean;\n  isSavedChatsEnabled: () => boolean;\n\n  setSavedChatsErrorStatus: (hasError: boolean) => void;\n  updateSavedChatsList: (chats: SavedChat[]) => void;\n  generateAuditEvent: (type: string, details: Record<string, any>, id?: string) => Promise<void>;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class SavedChatsService {\n  private operationConfig: SavedChatsOperationConfig | undefined;\n\n  constructor() {}\n\n  public init(config: SavedChatsOperationConfig): void {\n    if (this.operationConfig) {\n      console.warn('SavedChatsService already initialized.');\n      return;\n    }\n    this.operationConfig = config;\n  }\n\n  private ensureInitialized(): void {\n    if (!this.operationConfig) {\n      throw new Error('SavedChatsService not initialized. Call init() first.');\n    }\n  }\n\n  public listChats(): void {\n    this.ensureInitialized();\n    const config = this.operationConfig!;\n\n    if (!config.isSavedChatsEnabled()) {\n      throwError(() => new Error(\"Operations on saved chats are disabled\"));\n    }\n\n    const data = {\n      action: \"SavedChatList\",\n      appName: config.getAppName(),\n      instanceId: config.getInstanceId(),\n      debug: config.getDebugFlag()\n    };\n    const searchParams = new URLSearchParams(Object.entries(data).reduce((acc, [key, value]) => {\n      acc[key] = String(value);\n      return acc;\n    }, {} as Record<string, string>));\n\n    from(get<{ savedChats: SavedChat[] }>(`plugin/${config.getRestUrl()}`, searchParams)).pipe(\n      tap(res => {\n        config.updateSavedChatsList(res.savedChats);\n        config.setSavedChatsErrorStatus(false);\n      }),\n      catchError((error) => {\n        const errorMsg = 'Error occurred while calling the SavedChatList API: ' + error.error?.errorMessage;\n        console.error('SavedChatsService: ' + errorMsg);\n        config.setSavedChatsErrorStatus(true);\n        return throwError(() => error);\n      })\n    ).subscribe();\n  }\n\n  public getChatById(id: string): Observable<SavedChatHistory | undefined> {\n    this.ensureInitialized();\n    const config = this.operationConfig!;\n\n    if (!config.isSavedChatsEnabled()) {\n      return throwError(() => new Error(\"Operations on saved chats are disabled\"));\n    }\n\n    const data = {\n      action: \"SavedChatGet\",\n      appName: config.getAppName(),\n      instanceId: config.getInstanceId(),\n      chatId: id,\n      debug: config.getDebugFlag()\n    };\n    const searchParams = new URLSearchParams(Object.entries(data).reduce((acc, [key, value]) => {\n      acc[key] = String(value);\n      return acc;\n    }, {} as Record<string, string>));\n\n    return from(get<{ savedChat: SavedChatHistory, executionTimeMilliseconds: number }>(`plugin/${config.getRestUrl()}`, searchParams)).pipe(\n      tap(res => {\n        config.generateAuditEvent('ast-saved-chat.load', { duration: res.executionTimeMilliseconds }, res.savedChat.id);\n      }),\n      map(res => res.savedChat),\n      catchError((error) => {\n        const errorMsg = 'Error occurred while calling the SavedChatGet API: ' + error.error?.errorMessage;\n        console.error('SavedChatsService: ' + errorMsg);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  public addChat(id: string, messages: ChatMessage[]): Observable<SavedChatResponse> {\n    this.ensureInitialized();\n    const config = this.operationConfig!;\n\n    if (!config.isSavedChatsEnabled()) {\n      return throwError(() => new Error(\"Operations on saved chats are disabled\"));\n    }\n\n    const data = {\n      action: \"SavedChatAdd\",\n      appName: config.getAppName(),\n      instanceId: config.getInstanceId(),\n      chatId: id,\n      history: messages,\n      debug: config.getDebugFlag()\n    };\n\n    return from(post<SavedChatResponse>(`plugin/${config.getRestUrl()}`, data)).pipe(\n      tap(res => {\n        config.generateAuditEvent('ast-saved-chat.add', { duration: res.executionTimeMilliseconds }, res.savedChat.id);\n      }),\n      catchError((error) => {\n        const errorMsg = 'Error occurred while calling the SavedChatAdd API: ' + error.error?.errorMessage;\n        console.error('SavedChatsService: ' + errorMsg);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  public updateChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChatResponse> {\n    this.ensureInitialized();\n    const config = this.operationConfig!;\n\n    if (!config.isSavedChatsEnabled()) {\n      return throwError(() => new Error(\"Operations on saved chats are disabled\"));\n    }\n\n    const data: any = {\n      action: \"SavedChatUpdate\",\n      appName: config.getAppName(),\n      instanceId: config.getInstanceId(),\n      chatId: id,\n      debug: config.getDebugFlag()\n    };\n\n    if (name) data[\"title\"] = name;\n    if (messages) data[\"history\"] = messages;\n\n    return from(post<SavedChatResponse>(`plugin/${config.getRestUrl()}`, data)).pipe(\n      catchError((error) => {\n        const errorMsg = 'Error occurred while calling the SavedChatUpdate API: ' + error.error?.errorMessage;\n        console.error('SavedChatsService: ' + errorMsg);\n        return throwError(() => error);\n      })\n    );\n  }\n\n  public deleteChat(ids: string[]): Observable<DeleteSavedChatResponse> {\n    this.ensureInitialized();\n    const config = this.operationConfig!;\n\n    if (!config.isSavedChatsEnabled()) {\n      return throwError(() => new Error(\"Operations on saved chats are disabled\"));\n    }\n\n    const data = {\n      action: \"SavedChatDelete\",\n      appName: config.getAppName(),\n      instanceId: config.getInstanceId(),\n      chatIds: ids,\n      debug: config.getDebugFlag()\n    };\n\n    return from(post<DeleteSavedChatResponse>(`plugin/${config.getRestUrl()}`, data)).pipe(\n      catchError((error) => {\n        return throwError(() => error);\n      })\n    );\n  }\n\n  public isExistingChat(id: string): Observable<boolean> {\n    this.ensureInitialized();\n    const config = this.operationConfig!;\n\n    if (!config.isSavedChatsEnabled()) {\n      return throwError(() => new Error(\"Operations on saved chats are disabled\"));\n    }\n\n    const data = {\n      action: \"SavedChatExist\",\n      appName: config.getAppName(),\n      instanceId: config.getInstanceId(),\n      chatId: id,\n      debug: config.getDebugFlag()\n    };\n    const searchParams = new URLSearchParams(Object.entries(data).reduce((acc, [key, value]) => {\n      acc[key] = String(value);\n      return acc;\n    }, {} as Record<string, string>));\n\n    return from(get<{ exists: boolean, executionTime: string, executionTimeMilliseconds: number }>(`plugin/${config.getRestUrl()}`, searchParams)).pipe(\n      map(res => res.exists),\n      catchError((error) => {\n        const errorMsg = 'Error occurred while calling the SavedChatExist API: ' + error.error?.errorMessage;\n        console.error('SavedChatsService: ' + errorMsg);\n        return throwError(() => error);\n      })\n    );\n  }\n}\n"]}
|
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
|
+
import { TranslocoService } from '@jsverse/transloco';
|
|
3
|
+
import { fetchApp, fetchPrincipal, fetchUserSettings, sha512, } from '@sinequa/atomic';
|
|
4
|
+
import { DialogUpdatesComponent } from '../dialogs/updates.component';
|
|
5
|
+
import { AppService } from './app.service';
|
|
6
|
+
import { DialogService } from './dialog.service';
|
|
7
|
+
import { NotificationsService } from './notification.service';
|
|
8
|
+
import { PrincipalService } from './principal.service';
|
|
9
|
+
import { UserSettingsWebService } from './user-settings.service';
|
|
10
|
+
import { chatConfigSchema } from '../types';
|
|
11
|
+
import { getAssistantJsonFromCCApp } from '../utils/assistant-json';
|
|
12
|
+
import * as i0 from "@angular/core";
|
|
13
|
+
export class AssistantConfigurationService {
|
|
14
|
+
constructor() {
|
|
15
|
+
this.appService = inject(AppService);
|
|
16
|
+
this.userSettingsService = inject(UserSettingsWebService);
|
|
17
|
+
this.principalService = inject(PrincipalService);
|
|
18
|
+
this.modalService = inject(DialogService);
|
|
19
|
+
this.notificationsService = inject(NotificationsService);
|
|
20
|
+
this.transloco = inject(TranslocoService);
|
|
21
|
+
}
|
|
22
|
+
init(context) {
|
|
23
|
+
if (this.context) {
|
|
24
|
+
console.warn('AssistantConfigurationService already initialized.');
|
|
25
|
+
return;
|
|
26
|
+
}
|
|
27
|
+
this.context = context;
|
|
28
|
+
}
|
|
29
|
+
ensureInitialized() {
|
|
30
|
+
if (!this.context) {
|
|
31
|
+
throw new Error('AssistantConfigurationService not initialized. Call init() first.');
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
getAssistantsSetting() {
|
|
35
|
+
if (!this.userSettingsService.userSettings) {
|
|
36
|
+
this.userSettingsService.userSettings = {};
|
|
37
|
+
}
|
|
38
|
+
if (!this.userSettingsService.userSettings["assistants"]) {
|
|
39
|
+
this.userSettingsService.userSettings["assistants"] = {};
|
|
40
|
+
}
|
|
41
|
+
return this.userSettingsService.userSettings["assistants"];
|
|
42
|
+
}
|
|
43
|
+
async initChatConfig() {
|
|
44
|
+
this.ensureInitialized();
|
|
45
|
+
const context = this.context;
|
|
46
|
+
// fetch the standard app config to get the defaultValues of the chat config for the given instance
|
|
47
|
+
// Persist the app in the app service
|
|
48
|
+
const capp = await fetchApp();
|
|
49
|
+
this.appService.app = capp;
|
|
50
|
+
const settings = await fetchUserSettings();
|
|
51
|
+
this.userSettingsService.userSettings = settings;
|
|
52
|
+
const key = context.getChatInstanceId();
|
|
53
|
+
const userSettingsConfig = this.getAssistantsSetting()[key] || {};
|
|
54
|
+
const principal = await fetchPrincipal();
|
|
55
|
+
this.principalService.principal = principal;
|
|
56
|
+
const standardChatConfig = getAssistantJsonFromCCApp(this.appService.app, key);
|
|
57
|
+
try {
|
|
58
|
+
// Validate the whole config object against the schema
|
|
59
|
+
chatConfigSchema.parse(standardChatConfig);
|
|
60
|
+
// If the user preferences do not contain a config's defaultValues object, keep using the standard app config and nothing to store in the user preferences
|
|
61
|
+
if (!userSettingsConfig.defaultValues) {
|
|
62
|
+
context.setAssistantConfig({ ...standardChatConfig });
|
|
63
|
+
context.setInitConfigStatus(true);
|
|
64
|
+
}
|
|
65
|
+
else { // If the user has its own defaultValues in its userSettings, then we need to check for potential updates made by admins in the meantime and how he wants to manage them
|
|
66
|
+
// Retrieve already stored hashes in the user settings if exists
|
|
67
|
+
const appliedDefaultValuesHash = userSettingsConfig.hashes?.["applied-defaultValues-hash"];
|
|
68
|
+
const skippedDefaultValuesHash = userSettingsConfig.hashes?.["skipped-defaultValues-hash"];
|
|
69
|
+
// Create a hash of the current defaultValues of the standardChatConfig
|
|
70
|
+
const currentDefaultValuesHash = await sha512(JSON.stringify(standardChatConfig.defaultValues));
|
|
71
|
+
// Implement the tracking mechanism to notify the user about the available updates in the defaultValues object of the standard app config
|
|
72
|
+
const condition = (currentDefaultValuesHash !== appliedDefaultValuesHash) && (currentDefaultValuesHash !== skippedDefaultValuesHash);
|
|
73
|
+
if (condition) {
|
|
74
|
+
this.modalService.open(DialogUpdatesComponent)
|
|
75
|
+
.then(res => {
|
|
76
|
+
if (res === "dialog-confirm") {
|
|
77
|
+
const hashes = { ...userSettingsConfig.hashes, "applied-defaultValues-hash": currentDefaultValuesHash, "skipped-defaultValues-hash": undefined };
|
|
78
|
+
// Update the chat config and store its defaultValues in the user preferences
|
|
79
|
+
this.updateChatConfig({ ...standardChatConfig }, hashes, true);
|
|
80
|
+
context.setInitConfigStatus(true);
|
|
81
|
+
context.generateAuditEvent("ast-configuration.click", { 'configuration': JSON.stringify({ ...standardChatConfig }) });
|
|
82
|
+
}
|
|
83
|
+
else if (res === "dialog-no") {
|
|
84
|
+
// Do not notify the user about changes while this skipped version is not updated
|
|
85
|
+
const hashes = { ...userSettingsConfig.hashes, "skipped-defaultValues-hash": currentDefaultValuesHash };
|
|
86
|
+
this.updateChatConfig({ ...standardChatConfig, defaultValues: userSettingsConfig.defaultValues }, hashes, false);
|
|
87
|
+
context.setInitConfigStatus(true);
|
|
88
|
+
}
|
|
89
|
+
else {
|
|
90
|
+
// Just pick the version in the user settings, nothing to be updated
|
|
91
|
+
context.setAssistantConfig({ ...standardChatConfig, defaultValues: userSettingsConfig.defaultValues });
|
|
92
|
+
context.setInitConfigStatus(true);
|
|
93
|
+
}
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
else {
|
|
97
|
+
// No available updates Or updates has been already skipped, then just pick the version in the user settings
|
|
98
|
+
context.setAssistantConfig({ ...standardChatConfig, defaultValues: userSettingsConfig.defaultValues });
|
|
99
|
+
context.setInitConfigStatus(true);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
catch (error) {
|
|
104
|
+
this.notificationsService.error(`Missing valid configuration for the assistant instance '${key}'. See the browser console messages for details on the missing or incorrect properties.`);
|
|
105
|
+
// Ensure error is properly typed for accessing 'issues'
|
|
106
|
+
const issues = (error instanceof Error && 'issues' in error) ? error.issues : undefined;
|
|
107
|
+
throw new Error(`Missing valid configuration for the assistant instance '${key}' . \n ${JSON.stringify(issues, null, 2)}`);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
updateChatConfig(config, hashes, notify = true, successCallback, errorCallback) {
|
|
111
|
+
this.ensureInitialized();
|
|
112
|
+
const context = this.context;
|
|
113
|
+
context.setAssistantConfig(config);
|
|
114
|
+
const assistants = Object.assign({}, this.getAssistantsSetting());
|
|
115
|
+
const chatInstanceId = context.getChatInstanceId();
|
|
116
|
+
assistants[chatInstanceId] = { ...assistants[chatInstanceId], defaultValues: config.defaultValues };
|
|
117
|
+
if (hashes)
|
|
118
|
+
assistants[chatInstanceId].hashes = hashes;
|
|
119
|
+
this.userSettingsService.patch({ assistants }).subscribe(() => {
|
|
120
|
+
if (notify) {
|
|
121
|
+
if (successCallback) {
|
|
122
|
+
successCallback();
|
|
123
|
+
}
|
|
124
|
+
else {
|
|
125
|
+
this.notificationsService.success(this.transloco.translate('chat.saveChatConfig.success', { value: chatInstanceId }));
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
}, error => {
|
|
129
|
+
if (notify) {
|
|
130
|
+
if (errorCallback) {
|
|
131
|
+
errorCallback();
|
|
132
|
+
}
|
|
133
|
+
else {
|
|
134
|
+
this.notificationsService.error(this.transloco.translate('chat.saveChatConfig.fail', { value: chatInstanceId }));
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
console.error("Could not patch assistants!", error);
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
getWSRequestsUrl() {
|
|
141
|
+
this.ensureInitialized();
|
|
142
|
+
const context = this.context;
|
|
143
|
+
const url = context.getAssistantConfigValue()?.connectionSettings.websocketEndpoint;
|
|
144
|
+
if (url) {
|
|
145
|
+
context.setWSRequestsUrl(url);
|
|
146
|
+
}
|
|
147
|
+
else {
|
|
148
|
+
throw new Error(`The property 'websocketEndpoint' must be provided`);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
getRESTRequestsUrl() {
|
|
152
|
+
this.ensureInitialized();
|
|
153
|
+
const context = this.context;
|
|
154
|
+
const url = context.getAssistantConfigValue()?.connectionSettings.restEndpoint;
|
|
155
|
+
if (url) {
|
|
156
|
+
context.setRESTRequestsUrl(url);
|
|
157
|
+
}
|
|
158
|
+
else {
|
|
159
|
+
throw new Error(`The property 'restEndpoint' must be provided`);
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AssistantConfigurationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
163
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AssistantConfigurationService, providedIn: 'root' }); }
|
|
164
|
+
}
|
|
165
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AssistantConfigurationService, decorators: [{
|
|
166
|
+
type: Injectable,
|
|
167
|
+
args: [{
|
|
168
|
+
providedIn: 'root'
|
|
169
|
+
}]
|
|
170
|
+
}], ctorParameters: () => [] });
|
|
171
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"assistant-configuration.service.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/services/assistant-configuration.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { TranslocoService } from '@jsverse/transloco';\nimport {\n  fetchApp,\n  fetchPrincipal,\n  fetchUserSettings,\n  sha512,\n} from '@sinequa/atomic';\n\nimport { DialogUpdatesComponent } from '../dialogs/updates.component';\nimport { AppService } from './app.service';\nimport { DialogService } from './dialog.service';\nimport { NotificationsService } from './notification.service';\nimport { PrincipalService } from './principal.service';\nimport { UserSettingsWebService } from './user-settings.service';\nimport { ChatConfig, chatConfigSchema } from '../types';\nimport { getAssistantJsonFromCCApp } from '../utils/assistant-json';\n\nexport interface AssistantConfigurationOperationContext {\n  getChatInstanceId: () => string;\n  getChatId: () => string;\n  getAssistantConfigValue: () => ChatConfig | undefined;\n\n  setAssistantConfig: (config: ChatConfig | undefined) => void;\n  setWSRequestsUrl: (url: string) => void;\n  setRESTRequestsUrl: (url: string) => void;\n  setInitConfigStatus: (status: boolean) => void;\n  generateAuditEvent: (type: string, details: Record<string, any>, id?: string) => Promise<void>;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class AssistantConfigurationService {\n  private context: AssistantConfigurationOperationContext | undefined;\n\n  private appService = inject(AppService);\n  private userSettingsService = inject(UserSettingsWebService);\n  private principalService = inject(PrincipalService);\n  private modalService = inject(DialogService);\n  private notificationsService = inject(NotificationsService);\n  private transloco = inject(TranslocoService);\n\n  constructor() {}\n\n  public init(context: AssistantConfigurationOperationContext): void {\n    if (this.context) {\n      console.warn('AssistantConfigurationService already initialized.');\n      return;\n    }\n    this.context = context;\n  }\n\n  private ensureInitialized(): void {\n    if (!this.context) {\n      throw new Error('AssistantConfigurationService not initialized. Call init() first.');\n    }\n  }\n\n  public getAssistantsSetting(): any {\n    if (!this.userSettingsService.userSettings) {\n      this.userSettingsService.userSettings = {};\n    }\n    if (!this.userSettingsService.userSettings[\"assistants\"]) {\n      this.userSettingsService.userSettings[\"assistants\"] = {};\n    }\n    return this.userSettingsService.userSettings[\"assistants\"];\n  }\n\n  public async initChatConfig(): Promise<void> {\n    this.ensureInitialized();\n    const context = this.context!;\n\n    // fetch the standard app config to get the defaultValues of the chat config for the given instance\n    // Persist the app in the app service\n    const capp = await fetchApp();\n    this.appService.app = capp;\n\n    const settings = await fetchUserSettings<{}>();\n    this.userSettingsService.userSettings = settings;\n    const key = context.getChatInstanceId();\n    const userSettingsConfig = this.getAssistantsSetting()[key] || {};\n\n    const principal = await fetchPrincipal();\n    this.principalService.principal = principal;\n\n    const standardChatConfig = getAssistantJsonFromCCApp(this.appService.app, key);\n\n    try {\n      // Validate the whole config object against the schema\n      chatConfigSchema.parse(standardChatConfig);\n      // If the user preferences do not contain a config's defaultValues object, keep using the standard app config and nothing to store in the user preferences\n      if (!userSettingsConfig.defaultValues) {\n        context.setAssistantConfig({ ...standardChatConfig });\n        context.setInitConfigStatus(true);\n      } else {// If the user has its own defaultValues in its userSettings, then we need to check for potential updates made by admins in the meantime and how he wants to manage them\n\n        // Retrieve already stored hashes in the user settings if exists\n        const appliedDefaultValuesHash = userSettingsConfig.hashes?.[\"applied-defaultValues-hash\"];\n        const skippedDefaultValuesHash = userSettingsConfig.hashes?.[\"skipped-defaultValues-hash\"];\n        // Create a hash of the current defaultValues of the standardChatConfig\n        const currentDefaultValuesHash = await sha512(JSON.stringify(standardChatConfig.defaultValues));\n\n        // Implement the tracking mechanism to notify the user about the available updates in the defaultValues object of the standard app config\n        const condition = (currentDefaultValuesHash !== appliedDefaultValuesHash) && (currentDefaultValuesHash !== skippedDefaultValuesHash);\n        if (condition) {\n          this.modalService.open(DialogUpdatesComponent)\n            .then(res => {\n              if (res === \"dialog-confirm\") {\n                const hashes = { ...userSettingsConfig.hashes, \"applied-defaultValues-hash\": currentDefaultValuesHash, \"skipped-defaultValues-hash\": undefined };\n                // Update the chat config and store its defaultValues in the user preferences\n                this.updateChatConfig({ ...standardChatConfig }, hashes, true);\n                context.setInitConfigStatus(true);\n                context.generateAuditEvent(\"ast-configuration.click\", { 'configuration': JSON.stringify({ ...standardChatConfig }) });\n              } else if (res === \"dialog-no\") {\n                // Do not notify the user about changes while this skipped version is not updated\n                const hashes = { ...userSettingsConfig.hashes, \"skipped-defaultValues-hash\": currentDefaultValuesHash };\n                this.updateChatConfig({ ...standardChatConfig, defaultValues: userSettingsConfig.defaultValues }, hashes, false);\n                context.setInitConfigStatus(true);\n              } else {\n                // Just pick the version in the user settings, nothing to be updated\n                context.setAssistantConfig({ ...standardChatConfig, defaultValues: userSettingsConfig.defaultValues });\n                context.setInitConfigStatus(true);\n              }\n            });\n        } else {\n          // No available updates Or updates has been already skipped, then just pick the version in the user settings\n          context.setAssistantConfig({ ...standardChatConfig, defaultValues: userSettingsConfig.defaultValues });\n          context.setInitConfigStatus(true);\n        }\n      }\n    } catch (error) {\n      this.notificationsService.error(`Missing valid configuration for the assistant instance '${key}'. See the browser console messages for details on the missing or incorrect properties.`);\n      // Ensure error is properly typed for accessing 'issues'\n      const issues = (error instanceof Error && 'issues' in error) ? (error as any).issues : undefined;\n      throw new Error(`Missing valid configuration for the assistant instance '${key}' . \\n ${JSON.stringify(issues, null, 2)}`);\n    }\n  }\n\n  public updateChatConfig(\n    config: ChatConfig,\n    hashes?: { \"applied-defaultValues-hash\"?: string, \"skipped-defaultValues-hash\"?: string },\n    notify = true,\n    successCallback?: () => any,\n    errorCallback?: () => any\n  ): void {\n    this.ensureInitialized();\n    const context = this.context!;\n\n    context.setAssistantConfig(config);\n\n    const assistants = Object.assign({}, this.getAssistantsSetting());\n    const chatInstanceId = context.getChatInstanceId();\n    assistants[chatInstanceId] = { ...assistants[chatInstanceId], defaultValues: config.defaultValues };\n    if (hashes) assistants[chatInstanceId].hashes = hashes;\n\n    this.userSettingsService.patch({ assistants }).subscribe(\n      () => {\n        if (notify) {\n          if (successCallback) {\n            successCallback();\n          } else {\n            this.notificationsService.success(this.transloco.translate('chat.saveChatConfig.success', { value: chatInstanceId }));\n          }\n        }\n      },\n      error => {\n        if (notify) {\n          if (errorCallback) {\n            errorCallback();\n          } else {\n            this.notificationsService.error(this.transloco.translate('chat.saveChatConfig.fail', { value: chatInstanceId }));\n          }\n        }\n        console.error(\"Could not patch assistants!\", error);\n      }\n    );\n  }\n\n  public getWSRequestsUrl() {\n    this.ensureInitialized();\n    const context = this.context!;\n    const url = context.getAssistantConfigValue()?.connectionSettings.websocketEndpoint;\n    if (url) {\n      context.setWSRequestsUrl(url);\n    } else {\n      throw new Error(`The property 'websocketEndpoint' must be provided`);\n    }\n  }\n\n  public getRESTRequestsUrl() {\n    this.ensureInitialized();\n    const context = this.context!;\n    const url = context.getAssistantConfigValue()?.connectionSettings.restEndpoint;\n    if (url) {\n      context.setRESTRequestsUrl(url);\n    } else {\n      throw new Error(`The property 'restEndpoint' must be provided`);\n    }\n  }\n}\n"]}
|