@sinequa/assistant 3.7.8 → 3.8.1

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.
Files changed (32) hide show
  1. package/chat/chat-settings-v3/chat-settings-v3.component.d.ts +6 -5
  2. package/chat/chat.component.d.ts +7 -15
  3. package/chat/chat.service.d.ts +5 -5
  4. package/chat/documents-upload/document-list/document-list.component.d.ts +77 -0
  5. package/chat/documents-upload/document-overview/document-overview.component.d.ts +41 -0
  6. package/chat/documents-upload/document-upload/document-upload.component.d.ts +98 -0
  7. package/chat/documents-upload/documents-upload.model.d.ts +66 -0
  8. package/chat/documents-upload/documents-upload.service.d.ts +174 -0
  9. package/chat/public-api.d.ts +5 -0
  10. package/chat/styles/assistant.scss +2 -0
  11. package/chat/types.d.ts +76 -20
  12. package/chat/websocket-chat.service.d.ts +1 -1
  13. package/esm2020/chat/chat-message/chat-message.component.mjs +3 -3
  14. package/esm2020/chat/chat-reference/chat-reference.component.mjs +3 -3
  15. package/esm2020/chat/chat-settings-v3/chat-settings-v3.component.mjs +13 -10
  16. package/esm2020/chat/chat.component.mjs +31 -34
  17. package/esm2020/chat/chat.service.mjs +9 -9
  18. package/esm2020/chat/documents-upload/document-list/document-list.component.mjs +191 -0
  19. package/esm2020/chat/documents-upload/document-overview/document-overview.component.mjs +80 -0
  20. package/esm2020/chat/documents-upload/document-upload/document-upload.component.mjs +258 -0
  21. package/esm2020/chat/documents-upload/documents-upload.model.mjs +2 -0
  22. package/esm2020/chat/documents-upload/documents-upload.service.mjs +289 -0
  23. package/esm2020/chat/public-api.mjs +6 -1
  24. package/esm2020/chat/saved-chats/saved-chats.component.mjs +4 -4
  25. package/esm2020/chat/token-progress-bar/token-progress-bar.component.mjs +3 -3
  26. package/esm2020/chat/types.mjs +17 -6
  27. package/esm2020/chat/websocket-chat.service.mjs +2 -2
  28. package/fesm2015/sinequa-assistant-chat.mjs +3648 -2864
  29. package/fesm2015/sinequa-assistant-chat.mjs.map +1 -1
  30. package/fesm2020/sinequa-assistant-chat.mjs +3626 -2843
  31. package/fesm2020/sinequa-assistant-chat.mjs.map +1 -1
  32. package/package.json +4 -2
@@ -0,0 +1,258 @@
1
+ import { CommonModule } from "@angular/common";
2
+ import { Component, inject, Input, ViewChild, } from "@angular/core";
3
+ import { NgxFlowModule } from "@flowjs/ngx-flow";
4
+ import { BehaviorSubject, catchError, EMPTY, filter, finalize, interval, Subscription, switchMap, takeWhile, tap, startWith, } from "rxjs";
5
+ import { LoginService } from "@sinequa/core/login";
6
+ import { NotificationsService } from "@sinequa/core/notification";
7
+ import { UtilsModule } from "@sinequa/components/utils";
8
+ import { InstanceManagerService } from "../../instance-manager.service";
9
+ import { DocumentsUploadService } from "../documents-upload.service";
10
+ import { FormatIconComponent } from '../../format-icon/format-icon.component';
11
+ import * as i0 from "@angular/core";
12
+ import * as i1 from "@angular/common";
13
+ import * as i2 from "@flowjs/ngx-flow";
14
+ import * as i3 from "@sinequa/components/utils";
15
+ export class DocumentUploadComponent {
16
+ constructor() {
17
+ /** Polling interval in milliseconds to update the indexing status of the uploaded documents */
18
+ this.pollingInterval = 1000;
19
+ this.flowConfig = {
20
+ // Disables chunk testing before uploading actual file data. Thus, avoids unnecessary requests and speeds up uploads
21
+ testChunks: false,
22
+ // Allows multiple file uploads simultaneously
23
+ singleFile: false,
24
+ // Allows the same file to be uploaded multiple times
25
+ allowDuplicateUploads: true,
26
+ query: () => {
27
+ return {}; // Empty object to prevent query parameters
28
+ },
29
+ preprocess: (chunk) => chunk.abort() // Prevents the default flow upload of chunks
30
+ };
31
+ this.errorAlerts = [];
32
+ this.dragging$ = new BehaviorSubject(false);
33
+ this.indexing$ = new BehaviorSubject(false);
34
+ this.indexingInfos$ = new BehaviorSubject(undefined);
35
+ this.uploading$ = new BehaviorSubject(false);
36
+ this.uploadingInfos$ = new BehaviorSubject(undefined);
37
+ this._subscription = new Subscription();
38
+ this.loginService = inject(LoginService);
39
+ this.instanceManagerService = inject(InstanceManagerService);
40
+ this.documentsUploadService = inject(DocumentsUploadService);
41
+ this.notificationsService = inject(NotificationsService);
42
+ }
43
+ ngOnInit() {
44
+ this._subscription.add(this.loginService.events
45
+ .pipe(filter((e) => e.type === "login-complete"), tap((_) => this.instantiateChatService()), switchMap((_) => this.chatService.assistantConfig$), filter((config) => !!config), tap((_) => this.documentsUploadService.init(this.chatService)), catchError((error) => {
46
+ console.error(error);
47
+ this.notificationsService.error(error);
48
+ return EMPTY;
49
+ }))
50
+ .subscribe());
51
+ this._subscription.add(this.flow.events$.subscribe((event) => {
52
+ const evt = event;
53
+ // Kept it just for reference to show how to access data of different events
54
+ if (event.type === "filesAdded") {
55
+ const addedFiles = evt.event[0];
56
+ this._onFilesAdded(addedFiles);
57
+ }
58
+ if (event.type === "filesSubmitted") {
59
+ this._onFilesSubmitted();
60
+ }
61
+ }));
62
+ // Override the upload method of the flow directive
63
+ this.flow.upload = this.startUpload.bind(this);
64
+ }
65
+ ngOnDestroy() {
66
+ this._subscription.unsubscribe();
67
+ }
68
+ instantiateChatService() {
69
+ this.chatService = this.instanceManagerService.getInstance(this.instanceId);
70
+ }
71
+ /**
72
+ * Handles the submission of files.
73
+ *
74
+ * @remarks
75
+ * This method performs the following checks on the submitted files:
76
+ *
77
+ * 1. Checks if the number of files submitted exceeds the remainingFileCount defined in the constraints.
78
+ * - If the number of files exceeds the allowed count, all files are removed from the flow.
79
+ * - An error message is added to the error alerts.
80
+ *
81
+ * 2. Checks if the total size of the files submitted exceeds the remainingFileSize defined in the constraints.
82
+ * - If the total size exceeds the allowed size, all files are removed from the flow.
83
+ * - An error message is added to the error alerts.
84
+ *
85
+ * 3. Checks if the file extension of the files submitted is not allowed.
86
+ * - If the file extension is not allowed, the file is removed from the flow.
87
+ * - An error message is added to the error alerts.
88
+ */
89
+ _onFilesSubmitted() {
90
+ // Get all files from the flow
91
+ const files = [...this.flow.flowJs.files];
92
+ // Clear the error alerts
93
+ this.errorAlerts = [];
94
+ const errors = [];
95
+ /**
96
+ * Checks if the number of files submitted exceeds the remainingFileCount defined in the constraints.
97
+ * If the number of files exceeds the allowed count, all files are removed from the flow
98
+ * An error message is added to the error alerts.
99
+ */
100
+ if (files.length > this.documentsUploadService.uploadConfig$.value.constraints.remainingFileCount) {
101
+ for (const file of files) {
102
+ this.flow.flowJs.removeFile(file);
103
+ }
104
+ errors.push(`The number of files exceeds the allowed limit. You can upload up to ${this.documentsUploadService.uploadConfig$.value.constraints.remainingFileCount} files.`);
105
+ }
106
+ /**
107
+ * Checks if the total size of the files submitted exceeds the remainingFileSize defined in the constraints.
108
+ * If the total size exceeds the allowed size, all files are removed from the flow
109
+ * An error message is added to the error alerts.
110
+ *
111
+ * @remarks
112
+ * The maximum size of the files that can be uploaded is temporary set to the maximum of 128 MB and the remainingFileSizeMB defined in the constraints.
113
+ * Waiting for the plugin to handle this kestrel issue
114
+ */
115
+ else if (this.documentsUploadService.convertBytesToMB(this.flow.flowJs.getSize()) > Math.min(128, this.documentsUploadService.uploadConfig$.value.constraints.remainingFileSizeMB)) {
116
+ for (const file of files) {
117
+ this.flow.flowJs.removeFile(file);
118
+ }
119
+ errors.push(`The total size of the files exceeds the allowed limit. You can upload files up to ${Math.min(128, this.documentsUploadService.uploadConfig$.value.constraints.remainingFileSizeMB)} MB.`);
120
+ }
121
+ /**
122
+ * Checks if the file extension of the files submitted is not allowed.
123
+ * If the file extension is not allowed, the file is removed from the flow
124
+ * An error message is added to the error alerts.
125
+ */
126
+ else {
127
+ for (const file of files) {
128
+ if (!this.documentsUploadService.uploadConfig$.value.constraints.supportedFileExtensions.includes(`${file.getExtension()}`)) {
129
+ this.flow.flowJs.removeFile(file);
130
+ errors.push(`The file extension "${file.getExtension()}" is not supported.`);
131
+ }
132
+ }
133
+ }
134
+ // Add unique error messages to the error alerts
135
+ this.errorAlerts = [...new Set(errors)];
136
+ }
137
+ _onFilesAdded(files) { }
138
+ onDragenter() {
139
+ this.dragging$.next(true);
140
+ }
141
+ onDragleave(event) {
142
+ if (!event.relatedTarget ||
143
+ !event.relatedTarget.closest(".dropzone")) {
144
+ this.dragging$.next(false);
145
+ }
146
+ }
147
+ onDrop() {
148
+ this.dragging$.next(false);
149
+ }
150
+ trackTransfer(transfer) {
151
+ return transfer.id;
152
+ }
153
+ /**
154
+ * Initiates the upload process by invoking the `upload` method
155
+ * of the `flow` instance.
156
+ * The `upload` method is overridden in the `ngOnInit` method to match the requirements of the assistant API.
157
+ */
158
+ upload() {
159
+ this.flow.upload();
160
+ }
161
+ /**
162
+ * Initiates the file upload process by preparing the form data,
163
+ * setting the uploading flag, and making an API call to upload the files.
164
+ *
165
+ * @remarks
166
+ * - Collects all files from the `flowJs` instance and appends them to a `FormData` object.
167
+ * - Subscribes to the upload process to handle progress updates, completion, and errors.
168
+ * - Triggers tracking the indexing process upon successful upload completion.
169
+ * - Cleans up the `flow` instance after the upload process is finalized.
170
+ */
171
+ startUpload() {
172
+ const formData = new FormData();
173
+ // Add all files with their original names
174
+ this.flow.flowJs.files.forEach((file) => {
175
+ formData.append(file.name, file.file);
176
+ });
177
+ // Clear the error alerts
178
+ this.errorAlerts = [];
179
+ // Set the uploading flag to true
180
+ this.uploading$.next(true);
181
+ // Make the API call to upload the files
182
+ this._subscription.add(this.documentsUploadService.uploadDocuments(formData).pipe(tap((event) => {
183
+ if (event.type === 'uploadProgress') {
184
+ this.uploadingInfos$.next(event);
185
+ }
186
+ if (event.type === 'response') {
187
+ this.trackIndexingProcess(event.body.statusToken);
188
+ }
189
+ }), catchError((err) => {
190
+ this.uploading$.next(false); // Set the uploading flag to false ONLY in case of an error. Otherwise, keep it true until the start of indexing process in order to prevent flickering of the UI
191
+ this.uploadingInfos$.next(undefined); // Clear the uploading infos
192
+ console.error(err);
193
+ this.notificationsService.error("Error uploading files");
194
+ return EMPTY;
195
+ }), finalize(() => {
196
+ // Clear the flow instance after the upload is complete or an error occurs
197
+ this.flow.cancel();
198
+ })).subscribe());
199
+ }
200
+ /**
201
+ * Tracks the indexing process for a single documents upload session by polling the API for its status.
202
+ *
203
+ * @param token - A unique token representing the single documents upload session.
204
+ *
205
+ * @remarks
206
+ * - Clears the uploading flags and sets the indexing flag to true before starting the process.
207
+ * - Polls the API every second to retrieve the current indexing status.
208
+ * - Updates the `indexingInfos` property with the latest status.
209
+ * - Stops polling when the indexing process is completed or an error occurs.
210
+ * - Cleans up by resetting the indexing flags and clearing the `indexingInfos` property when the process is finalized.
211
+ */
212
+ trackIndexingProcess(token) {
213
+ // Clear the uploading flags
214
+ this.uploading$.next(false);
215
+ this.uploadingInfos$.next(undefined);
216
+ // Set the indexing flag to true
217
+ this.indexing$.next(true);
218
+ // Combine immediate API call with interval-based polling
219
+ this._subscription.add(interval(this.pollingInterval).pipe(startWith(0), // Trigger the API call immediately
220
+ switchMap(() => this.documentsUploadService.getIndexingStatus(token)), tap((res) => this.indexingInfos$.next(res)), takeWhile((response) => !response.isCompleted, true), catchError((err) => {
221
+ console.error(err);
222
+ this.notificationsService.error("Error retrieving indexing status");
223
+ return EMPTY;
224
+ }), finalize(() => {
225
+ // Clear the indexing flags
226
+ this.indexing$.next(false);
227
+ this.indexingInfos$.next(undefined);
228
+ })).subscribe());
229
+ }
230
+ /**
231
+ * Calculates the indexing progress as a percentage of completed documents.
232
+ *
233
+ * @param infos - The indexing information containing the list of documents and their statuses.
234
+ * @returns The progress as a number between 0 and 1, where 0 indicates no progress and 1 indicates all documents are processed (processed means either indexed or errored).
235
+ * Returns 0 if the input is invalid or there are no documents.
236
+ */
237
+ getIndexingProgress(infos) {
238
+ if (!infos || !infos.docs)
239
+ return 0;
240
+ const completed = infos.docs.filter(doc => doc.status === "Indexed" || doc.status === "Error").length;
241
+ const total = infos.docs.length;
242
+ return completed / total;
243
+ }
244
+ }
245
+ DocumentUploadComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: DocumentUploadComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
246
+ DocumentUploadComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: DocumentUploadComponent, isStandalone: true, selector: "sq-document-upload", inputs: { instanceId: "instanceId", pollingInterval: "pollingInterval" }, providers: [DocumentsUploadService], viewQueries: [{ propertyName: "flow", first: true, predicate: ["flow"], descendants: true, static: true }], ngImport: i0, template: "<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n <input\n type=\"file\"\n flowButton\n [flow]=\"flow.flowJs\"\n multiple\n hidden\n #fileInput>\n <div\n flowDrop\n [flow]=\"flow.flowJs\"\n (dragenter)=\"onDragenter()\"\n (dragleave)=\"onDragleave($event)\"\n (drop)=\"onDrop()\"\n (click)=\"fileInput.click()\"\n class=\"dropzone\"\n [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n <i class=\"fas fa-cloud-upload-alt\"></i>\n <span>Drag and drop files here Or</span>\n <span class=\"text-orange\">Click to browse</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>Drop files here</span>\n </ng-template>\n </div>\n\n <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <span>\n Uploading {{ (flow.transfers$ | async)!.transfers.length }} files\n </span>\n <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n </ng-container>\n <ng-template #indexingState>\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n <span>Indexing {{ indexingInfo.docs.length }} files</span>\n <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n </ng-container>\n </ng-template>\n </div>\n\n <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n <li *ngFor=\"let error of errorAlerts\">\n {{ error }}\n </li>\n </ul>\n\n <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n <ul>\n <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n <span [sqTooltip]=\"transfer.name\">{{ transfer.name }}</span>\n <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [sqTooltip]=\"'Cancel'\"></i>\n </li>\n </ul>\n </div>\n\n <div class=\"d-flex mt-2\">\n <button\n type=\"button\"\n class=\"btn btn-light cancel-btn me-2\"\n (click)=\"flow.cancel()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n Cancel\n </button>\n <button\n type=\"button\"\n class=\"upload-btn\"\n (click)=\"upload()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n Upload\n </button>\n </div>\n</div>\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}@keyframes dash-move{0%{background-position:0 0}to{background-position:100% 0}}.file-upload-container{width:100%;position:relative;padding:20px;background-color:#fff}.file-upload-container .dropzone{display:flex;flex-direction:column;justify-content:center;align-items:center;height:100px;border:2px dashed var(--ast-primary-color, #005DA7);border-color:var(--ast-secondary-color, #FF732E) var(--ast-primary-color, #005DA7) var(--ast-primary-color, #005DA7) var(--ast-secondary-color, #FF732E);border-radius:5px;padding:10px;cursor:pointer;background-color:#fff;transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:#fff8f1;color:var(--ast-secondary-color, #FF732E);border-color:var(--ast-secondary-color, #FF732E)}.file-upload-container .dropzone i{font-size:x-large}.file-upload-container .dropzone span{margin:0;font-size:small}.file-upload-container .dropzone span.text-orange{color:var(--ast-secondary-color, #FF732E)}.file-upload-container .file-list h6{color:#a9a9a9}.file-upload-container .file-list ul{list-style-type:none;padding:0}.file-upload-container .file-list ul li{display:flex;align-items:center;padding:10px;background-color:var(--ast-primary-bg, #f2f8fe);margin-bottom:5px;border-radius:5px;font-size:small}.file-upload-container .file-list ul li span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.file-upload-container .file-list ul li i{cursor:pointer}.file-upload-container .file-list ul li i:hover{color:var(--ast-primary-color, #005DA7)}.file-upload-container .error-list{display:flex;flex-direction:column;list-style:disc;background:var(--ast-error-color, rgba(249, 58, 55, .7));color:#fff;border-radius:5px}.file-upload-container .error-list li{padding:3px}.file-upload-container .upload-btn{background:linear-gradient(to right,#1d4ed8,#ec4899,#f97316);color:#fff;border:none;padding:8px 16px;border-radius:5px;cursor:pointer;width:100%}.file-upload-container .upload-btn:hover{background:linear-gradient(to right,#1d4ed8cc,#ec4899cc,#f97316cc)}.file-upload-container .upload-btn[disabled]{opacity:.3;cursor:not-allowed}.file-upload-container .cancel-btn{cursor:pointer;pointer-events:unset;width:100%}.file-upload-container .cancel-btn[disabled]{opacity:.3;cursor:not-allowed}\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: "pipe", type: i1.AsyncPipe, name: "async" }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }, { kind: "ngmodule", type: NgxFlowModule }, { kind: "directive", type: i2.ButtonDirective, selector: "[flowButton]", inputs: ["flowDirectoryOnly", "flowAttributes", "flow"] }, { kind: "directive", type: i2.DropDirective, selector: "[flowDrop]", inputs: ["flow"], exportAs: ["flowDrop"] }, { kind: "directive", type: i2.FlowDirective, selector: "[flowConfig]", inputs: ["flowConfig"], exportAs: ["flow"] }, { kind: "component", type: FormatIconComponent, selector: "sq-format-icon", inputs: ["extension"] }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i3.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }] });
247
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: DocumentUploadComponent, decorators: [{
248
+ type: Component,
249
+ args: [{ selector: "sq-document-upload", standalone: true, providers: [DocumentsUploadService], imports: [CommonModule, NgxFlowModule, FormatIconComponent, UtilsModule], template: "<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n <input\n type=\"file\"\n flowButton\n [flow]=\"flow.flowJs\"\n multiple\n hidden\n #fileInput>\n <div\n flowDrop\n [flow]=\"flow.flowJs\"\n (dragenter)=\"onDragenter()\"\n (dragleave)=\"onDragleave($event)\"\n (drop)=\"onDrop()\"\n (click)=\"fileInput.click()\"\n class=\"dropzone\"\n [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n <i class=\"fas fa-cloud-upload-alt\"></i>\n <span>Drag and drop files here Or</span>\n <span class=\"text-orange\">Click to browse</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>Drop files here</span>\n </ng-template>\n </div>\n\n <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <span>\n Uploading {{ (flow.transfers$ | async)!.transfers.length }} files\n </span>\n <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n </ng-container>\n <ng-template #indexingState>\n <i class=\"fas fa-spinner fa-pulse\"></i>\n <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n <span>Indexing {{ indexingInfo.docs.length }} files</span>\n <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n </ng-container>\n </ng-template>\n </div>\n\n <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n <li *ngFor=\"let error of errorAlerts\">\n {{ error }}\n </li>\n </ul>\n\n <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n <ul>\n <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n <span [sqTooltip]=\"transfer.name\">{{ transfer.name }}</span>\n <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [sqTooltip]=\"'Cancel'\"></i>\n </li>\n </ul>\n </div>\n\n <div class=\"d-flex mt-2\">\n <button\n type=\"button\"\n class=\"btn btn-light cancel-btn me-2\"\n (click)=\"flow.cancel()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n Cancel\n </button>\n <button\n type=\"button\"\n class=\"upload-btn\"\n (click)=\"upload()\"\n [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n Upload\n </button>\n </div>\n</div>\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}@keyframes dash-move{0%{background-position:0 0}to{background-position:100% 0}}.file-upload-container{width:100%;position:relative;padding:20px;background-color:#fff}.file-upload-container .dropzone{display:flex;flex-direction:column;justify-content:center;align-items:center;height:100px;border:2px dashed var(--ast-primary-color, #005DA7);border-color:var(--ast-secondary-color, #FF732E) var(--ast-primary-color, #005DA7) var(--ast-primary-color, #005DA7) var(--ast-secondary-color, #FF732E);border-radius:5px;padding:10px;cursor:pointer;background-color:#fff;transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:#fff8f1;color:var(--ast-secondary-color, #FF732E);border-color:var(--ast-secondary-color, #FF732E)}.file-upload-container .dropzone i{font-size:x-large}.file-upload-container .dropzone span{margin:0;font-size:small}.file-upload-container .dropzone span.text-orange{color:var(--ast-secondary-color, #FF732E)}.file-upload-container .file-list h6{color:#a9a9a9}.file-upload-container .file-list ul{list-style-type:none;padding:0}.file-upload-container .file-list ul li{display:flex;align-items:center;padding:10px;background-color:var(--ast-primary-bg, #f2f8fe);margin-bottom:5px;border-radius:5px;font-size:small}.file-upload-container .file-list ul li span{flex-grow:1;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.file-upload-container .file-list ul li i{cursor:pointer}.file-upload-container .file-list ul li i:hover{color:var(--ast-primary-color, #005DA7)}.file-upload-container .error-list{display:flex;flex-direction:column;list-style:disc;background:var(--ast-error-color, rgba(249, 58, 55, .7));color:#fff;border-radius:5px}.file-upload-container .error-list li{padding:3px}.file-upload-container .upload-btn{background:linear-gradient(to right,#1d4ed8,#ec4899,#f97316);color:#fff;border:none;padding:8px 16px;border-radius:5px;cursor:pointer;width:100%}.file-upload-container .upload-btn:hover{background:linear-gradient(to right,#1d4ed8cc,#ec4899cc,#f97316cc)}.file-upload-container .upload-btn[disabled]{opacity:.3;cursor:not-allowed}.file-upload-container .cancel-btn{cursor:pointer;pointer-events:unset;width:100%}.file-upload-container .cancel-btn[disabled]{opacity:.3;cursor:not-allowed}\n"] }]
250
+ }], propDecorators: { instanceId: [{
251
+ type: Input
252
+ }], pollingInterval: [{
253
+ type: Input
254
+ }], flow: [{
255
+ type: ViewChild,
256
+ args: ["flow", { static: true }]
257
+ }] } });
258
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"document-upload.component.js","sourceRoot":"","sources":["../../../../../../projects/assistant/chat/documents-upload/document-upload/document-upload.component.ts","../../../../../../projects/assistant/chat/documents-upload/document-upload/document-upload.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,SAAS,EACT,MAAM,EACN,KAAK,EAGL,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,aAAa,EAA2B,MAAM,kBAAkB,CAAC;AAE1E,OAAO,EACL,eAAe,EACf,UAAU,EACV,KAAK,EACL,MAAM,EACN,QAAQ,EACR,QAAQ,EACR,YAAY,EACZ,SAAS,EACT,SAAS,EACT,GAAG,EACH,SAAS,GACV,MAAM,MAAM,CAAC;AACd,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,oBAAoB,EAAE,MAAM,4BAA4B,CAAC;AAClE,OAAO,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;AACxD,OAAO,EAAE,sBAAsB,EAAE,MAAM,gCAAgC,CAAC;AAExE,OAAO,EAAE,sBAAsB,EAAE,MAAM,6BAA6B,CAAC;AACrE,OAAO,EAAE,mBAAmB,EAAE,MAAM,yCAAyC,CAAC;;;;;AAY9E,MAAM,OAAO,uBAAuB;IARpC;QAWE,+FAA+F;QACtF,oBAAe,GAAY,IAAI,CAAC;QAKhC,eAAU,GAAuB;YACxC,oHAAoH;YACpH,UAAU,EAAE,KAAK;YACjB,8CAA8C;YAC9C,UAAU,EAAE,KAAK;YACjB,qDAAqD;YACrD,qBAAqB,EAAE,IAAI;YAC3B,KAAK,EAAE,GAAG,EAAE;gBACV,OAAO,EAAE,CAAC,CAAC,2CAA2C;YACxD,CAAC;YACD,UAAU,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,6CAA6C;SACnF,CAAC;QACK,gBAAW,GAAa,EAAE,CAAC;QAC3B,cAAS,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAChD,cAAS,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAChD,mBAAc,GAAG,IAAI,eAAe,CAA4B,SAAS,CAAC,CAAC;QAC3E,eAAU,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QACjD,oBAAe,GAAG,IAAI,eAAe,CAA6B,SAAS,CAAC,CAAC;QAE5E,kBAAa,GAAG,IAAI,YAAY,EAAE,CAAC;QAEpC,iBAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QACpC,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,yBAAoB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;KA6P5D;IA3PC,QAAQ;QACN,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,YAAY,CAAC,MAAM;aACrB,IAAI,CACH,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,gBAAgB,CAAC,EAC1C,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,sBAAsB,EAAE,CAAC,EACzC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,gBAAgB,CAAC,EACnD,MAAM,CAAC,CAAC,MAA8B,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,EACpD,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,EAC9D,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;YACnB,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACrB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACvC,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,CACH;aACA,SAAS,EAAE,CACf,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YACpC,MAAM,GAAG,GAAG,KAAY,CAAC;YACzB,4EAA4E;YAC5E,IAAI,KAAK,CAAC,IAAI,KAAK,YAAY,EAAE;gBAC/B,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAe,CAAC;gBAC9C,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;aAChC;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,gBAAgB,EAAE;gBACnC,IAAI,CAAC,iBAAiB,EAAE,CAAC;aAC1B;QACH,CAAC,CAAC,CACH,CAAC;QAEF,mDAAmD;QACnD,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACjD,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,WAAW,EAAE,CAAC;IACnC,CAAC;IAED,sBAAsB;QACpB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,sBAAsB,CAAC,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC9E,CAAC;IAED;;;;;;;;;;;;;;;;;OAiBG;IACK,iBAAiB;QACvB,8BAA8B;QAC9B,MAAM,KAAK,GAAG,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC1C,yBAAyB;QACzB,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;QACtB,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B;;;;WAIG;QACH,IAAI,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,kBAAkB,EAAE;YAClG,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE;gBACxB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;aACnC;YACD,MAAM,CAAC,IAAI,CAAC,uEAAuE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,kBAAkB,SAAS,CAAC,CAAA;SAC7K;QAED;;;;;;;;WAQG;aACE,IAAI,IAAI,CAAC,sBAAsB,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,EAAE;YACnL,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE;gBACxB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;aACnC;YACD,MAAM,CAAC,IAAI,CAAC,qFAAqF,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAA;SACxM;QAED;;;;WAIG;aACE;YACH,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE;gBACxB,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,uBAAuB,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC,EAAE;oBAC5H,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;oBAClC,MAAM,CAAC,IAAI,CAAC,uBAAuB,IAAI,CAAC,YAAY,EAAE,qBAAqB,CAAC,CAAC;iBAC9E;aACF;SACF;QAED,gDAAgD;QAChD,IAAI,CAAC,WAAW,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC;IAC1C,CAAC;IAEO,aAAa,CAAC,KAAiB,IAAG,CAAC;IAE3C,WAAW;QACT,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC5B,CAAC;IAED,WAAW,CAAC,KAAgB;QAC1B,IACE,CAAC,KAAK,CAAC,aAAa;YACpB,CAAE,KAAK,CAAC,aAA6B,CAAC,OAAO,CAAC,WAAW,CAAC,EAC1D;YACA,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;SAC5B;IACH,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC7B,CAAC;IAED,aAAa,CAAC,QAAkB;QAC9B,OAAO,QAAQ,CAAC,EAAE,CAAC;IACrB,CAAC;IAGD;;;;OAIG;IACH,MAAM;QACJ,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;IACrB,CAAC;IAED;;;;;;;;;OASG;IACK,WAAW;QACjB,MAAM,QAAQ,GAAG,IAAI,QAAQ,EAAE,CAAC;QAEhC,0CAA0C;QAC1C,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;YACtC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;QAEH,yBAAyB;QACzB,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;QACtB,iCAAiC;QACjC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE3B,wCAAwC;QACxC,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,sBAAsB,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC,IAAI,CACxD,GAAG,CAAC,CAAC,KAAkB,EAAE,EAAE;YACzB,IAAI,KAAK,CAAC,IAAI,KAAK,gBAAgB,EAAE;gBACnC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAuB,CAAC,CAAC;aACpD;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,UAAU,EAAE;gBAC7B,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;aACnD;QACH,CAAC,CAAC,EACF,UAAU,CAAC,CAAC,GAAG,EAAE,EAAE;YACjB,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,iKAAiK;YAC9L,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,4BAA4B;YAClE,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACnB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC;YACzD,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,EACF,QAAQ,CAAC,GAAG,EAAE;YACZ,0EAA0E;YAC1E,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;QACrB,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CACd,CAAC;IACJ,CAAC;IAED;;;;;;;;;;;OAWG;IACH,oBAAoB,CAAC,KAAa;QAChC,4BAA4B;QAC5B,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5B,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACrC,gCAAgC;QAChC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE1B,yDAAyD;QACzD,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,QAAQ,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,IAAI,CACjC,SAAS,CAAC,CAAC,CAAC,EAAE,mCAAmC;QACjD,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,sBAAsB,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,EACrE,GAAG,CAAC,CAAC,GAAkB,EAAE,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAC1D,SAAS,CAAC,CAAC,QAAuB,EAAE,EAAE,CAAC,CAAC,QAAQ,CAAC,WAAW,EAAE,IAAI,CAAC,EACnE,UAAU,CAAC,CAAC,GAAG,EAAE,EAAE;YACjB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACnB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,kCAAkC,CAAC,CAAC;YACpE,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,EACF,QAAQ,CAAC,GAAG,EAAE;YACZ,2BAA2B;YAC3B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACtC,CAAC,CAAC,CACH,CAAC,SAAS,EAAE,CACd,CAAC;IACJ,CAAC;IAED;;;;;;OAMG;IACH,mBAAmB,CAAC,KAAgC;QAClD,IAAI,CAAC,KAAK,IAAI,CAAC,KAAK,CAAC,IAAI;YAAE,OAAO,CAAC,CAAC;QACpC,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,MAAM,KAAK,SAAS,IAAI,GAAG,CAAC,MAAM,KAAK,OAAO,CAAC,CAAC,MAAM,CAAC;QACtG,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC;QAChC,OAAO,SAAS,GAAC,KAAK,CAAC;IACzB,CAAC;;oHA5RU,uBAAuB;wGAAvB,uBAAuB,2IAHvB,CAAC,sBAAsB,CAAC,sICvCrC,q9FA+EA,ypHDvCY,YAAY,0cAAE,aAAa,yYAAE,mBAAmB,iFAAE,WAAW;2FAE5D,uBAAuB;kBARnC,SAAS;+BACE,oBAAoB,cAGlB,IAAI,aACL,CAAC,sBAAsB,CAAC,WAC1B,CAAC,YAAY,EAAE,aAAa,EAAE,mBAAmB,EAAE,WAAW,CAAC;8BAI/D,UAAU;sBAAlB,KAAK;gBAEG,eAAe;sBAAvB,KAAK;gBAE+B,IAAI;sBAAxC,SAAS;uBAAC,MAAM,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE","sourcesContent":["import { CommonModule } from \"@angular/common\";\nimport {\n  Component,\n  inject,\n  Input,\n  OnDestroy,\n  OnInit,\n  ViewChild,\n} from \"@angular/core\";\nimport { NgxFlowModule, FlowDirective, Transfer } from \"@flowjs/ngx-flow\";\nimport { FlowFile } from \"flowjs\";\nimport {\n  BehaviorSubject,\n  catchError,\n  EMPTY,\n  filter,\n  finalize,\n  interval,\n  Subscription,\n  switchMap,\n  takeWhile,\n  tap,\n  startWith,\n} from \"rxjs\";\nimport { LoginService } from \"@sinequa/core/login\";\nimport { NotificationsService } from \"@sinequa/core/notification\";\nimport { UtilsModule } from \"@sinequa/components/utils\";\nimport { InstanceManagerService } from \"../../instance-manager.service\";\nimport { ChatService } from \"../../chat.service\";\nimport { DocumentsUploadService } from \"../documents-upload.service\";\nimport { FormatIconComponent } from '../../format-icon/format-icon.component';\nimport { ChatConfig } from \"../../types\";\nimport { IndexingInfos, UploadEvent, UploadingInfos } from \"../documents-upload.model\";\n\n@Component({\n  selector: \"sq-document-upload\",\n  templateUrl: \"./document-upload.component.html\",\n  styleUrls: [\"./document-upload.component.scss\"],\n  standalone: true,\n  providers: [DocumentsUploadService],\n  imports: [CommonModule, NgxFlowModule, FormatIconComponent, UtilsModule],\n})\nexport class DocumentUploadComponent implements OnInit, OnDestroy {\n  /** Define the key based on it, the appropriate chatService instance will be returned from instanceManagerService */\n  @Input() instanceId: string;\n  /** Polling interval in milliseconds to update the indexing status of the uploaded documents */\n  @Input() pollingInterval : number = 1000;\n  // Reference to the flow directive\n  @ViewChild(\"flow\", { static: true }) flow!: FlowDirective;\n\n  public chatService: ChatService;\n  readonly flowConfig: flowjs.FlowOptions = {\n    // Disables chunk testing before uploading actual file data. Thus, avoids unnecessary requests and speeds up uploads\n    testChunks: false,\n    // Allows multiple file uploads simultaneously\n    singleFile: false,\n    // Allows the same file to be uploaded multiple times\n    allowDuplicateUploads: true,\n    query: () => {\n      return {}; // Empty object to prevent query parameters\n    },\n    preprocess: (chunk) => chunk.abort() // Prevents the default flow upload of chunks\n  };\n  public errorAlerts: string[] = [];\n  public dragging$ = new BehaviorSubject<boolean>(false);\n  public indexing$ = new BehaviorSubject<boolean>(false);\n  public indexingInfos$ = new BehaviorSubject<IndexingInfos | undefined>(undefined);\n  public uploading$ = new BehaviorSubject<boolean>(false);\n  public uploadingInfos$ = new BehaviorSubject<UploadingInfos | undefined>(undefined);\n\n  private _subscription = new Subscription();\n\n  public loginService = inject(LoginService);\n  public instanceManagerService = inject(InstanceManagerService);\n  public documentsUploadService = inject(DocumentsUploadService);\n  public notificationsService = inject(NotificationsService);\n\n  ngOnInit(): void {\n    this._subscription.add(\n      this.loginService.events\n        .pipe(\n          filter((e) => e.type === \"login-complete\"),\n          tap((_) => this.instantiateChatService()),\n          switchMap((_) => this.chatService.assistantConfig$),\n          filter((config: ChatConfig | undefined) => !!config),\n          tap((_) => this.documentsUploadService.init(this.chatService)),\n          catchError((error) => {\n            console.error(error);\n            this.notificationsService.error(error);\n            return EMPTY;\n          })\n        )\n        .subscribe()\n    );\n\n    this._subscription.add(\n      this.flow.events$.subscribe((event) => {\n        const evt = event as any;\n        // Kept it just for reference to show how to access data of different events\n        if (event.type === \"filesAdded\") {\n          const addedFiles = evt.event[0] as FlowFile[];\n          this._onFilesAdded(addedFiles);\n        }\n        if (event.type === \"filesSubmitted\") {\n          this._onFilesSubmitted();\n        }\n      })\n    );\n\n    // Override the upload method of the flow directive\n    this.flow.upload = this.startUpload.bind(this);\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  /**\n   * Handles the submission of files.\n   *\n   * @remarks\n   * This method performs the following checks on the submitted files:\n   *\n   * 1. Checks if the number of files submitted exceeds the remainingFileCount defined in the constraints.\n   *    - If the number of files exceeds the allowed count, all files are removed from the flow.\n   *    - An error message is added to the error alerts.\n   *\n   * 2. Checks if the total size of the files submitted exceeds the remainingFileSize defined in the constraints.\n   *    - If the total size exceeds the allowed size, all files are removed from the flow.\n   *    - An error message is added to the error alerts.\n   *\n   * 3. Checks if the file extension of the files submitted is not allowed.\n   *    - If the file extension is not allowed, the file is removed from the flow.\n   *    - An error message is added to the error alerts.\n   */\n  private _onFilesSubmitted() {\n    // Get all files from the flow\n    const files = [...this.flow.flowJs.files];\n    // Clear the error alerts\n    this.errorAlerts = [];\n    const errors: string[] = [];\n\n    /**\n     * Checks if the number of files submitted exceeds the remainingFileCount defined in the constraints.\n     * If the number of files exceeds the allowed count, all files are removed from the flow\n     * An error message is added to the error alerts.\n     */\n    if (files.length > this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileCount) {\n      for (const file of files) {\n        this.flow.flowJs.removeFile(file);\n      }\n      errors.push(`The number of files exceeds the allowed limit. You can upload up to ${this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileCount} files.`)\n    }\n\n    /**\n     * Checks if the total size of the files submitted exceeds the remainingFileSize defined in the constraints.\n     * If the total size exceeds the allowed size, all files are removed from the flow\n     * An error message is added to the error alerts.\n     *\n     * @remarks\n     * The maximum size of the files that can be uploaded is temporary set to the maximum of 128 MB and the remainingFileSizeMB defined in the constraints.\n     * Waiting for the plugin to handle this kestrel issue\n     */\n    else if (this.documentsUploadService.convertBytesToMB(this.flow.flowJs.getSize()) > Math.min(128, this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileSizeMB)) {\n      for (const file of files) {\n        this.flow.flowJs.removeFile(file);\n      }\n      errors.push(`The total size of the files exceeds the allowed limit. You can upload files up to ${Math.min(128, this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileSizeMB)} MB.`)\n    }\n\n    /**\n     * Checks if the file extension of the files submitted is not allowed.\n     * If the file extension is not allowed, the file is removed from the flow\n     * An error message is added to the error alerts.\n     */\n    else {\n      for (const file of files) {\n        if (!this.documentsUploadService.uploadConfig$.value!.constraints.supportedFileExtensions.includes(`${file.getExtension()}`)) {\n          this.flow.flowJs.removeFile(file);\n          errors.push(`The file extension \"${file.getExtension()}\" is not supported.`);\n        }\n      }\n    }\n\n    // Add unique error messages to the error alerts\n    this.errorAlerts = [...new Set(errors)];\n  }\n\n  private _onFilesAdded(files: FlowFile[]) {}\n\n  onDragenter() {\n    this.dragging$.next(true);\n  }\n\n  onDragleave(event: DragEvent) {\n    if (\n      !event.relatedTarget ||\n      !(event.relatedTarget as HTMLElement).closest(\".dropzone\")\n    ) {\n      this.dragging$.next(false);\n    }\n  }\n\n  onDrop() {\n    this.dragging$.next(false);\n  }\n\n  trackTransfer(transfer: Transfer) {\n    return transfer.id;\n  }\n\n\n  /**\n   * Initiates the upload process by invoking the `upload` method\n   * of the `flow` instance.\n   * The `upload` method is overridden in the `ngOnInit` method to match the requirements of the assistant API.\n   */\n  upload() {\n    this.flow.upload();\n  }\n\n  /**\n   * Initiates the file upload process by preparing the form data,\n   * setting the uploading flag, and making an API call to upload the files.\n   *\n   * @remarks\n   * - Collects all files from the `flowJs` instance and appends them to a `FormData` object.\n   * - Subscribes to the upload process to handle progress updates, completion, and errors.\n   * - Triggers tracking the indexing process upon successful upload completion.\n   * - Cleans up the `flow` instance after the upload process is finalized.\n   */\n  private startUpload() {\n    const formData = new FormData();\n\n    // Add all files with their original names\n    this.flow.flowJs.files.forEach((file) => {\n      formData.append(file.name, file.file);\n    });\n\n    // Clear the error alerts\n    this.errorAlerts = [];\n    // Set the uploading flag to true\n    this.uploading$.next(true);\n\n    // Make the API call to upload the files\n    this._subscription.add(\n      this.documentsUploadService.uploadDocuments(formData).pipe(\n        tap((event: UploadEvent) => {\n          if (event.type === 'uploadProgress') {\n            this.uploadingInfos$.next(event as UploadingInfos);\n          }\n          if (event.type === 'response') {\n            this.trackIndexingProcess(event.body.statusToken);\n          }\n        }),\n        catchError((err) => {\n          this.uploading$.next(false); // Set the uploading flag to false ONLY in case of an error. Otherwise, keep it true until the start of indexing process in order to prevent flickering of the UI\n          this.uploadingInfos$.next(undefined); // Clear the uploading infos\n          console.error(err);\n          this.notificationsService.error(\"Error uploading files\");\n          return EMPTY;\n        }),\n        finalize(() => {\n          // Clear the flow instance after the upload is complete or an error occurs\n          this.flow.cancel();\n        })\n      ).subscribe()\n    );\n  }\n\n  /**\n   * Tracks the indexing process for a single documents upload session by polling the API for its status.\n   *\n   * @param token - A unique token representing the single documents upload session.\n   *\n   * @remarks\n   * - Clears the uploading flags and sets the indexing flag to true before starting the process.\n   * - Polls the API every second to retrieve the current indexing status.\n   * - Updates the `indexingInfos` property with the latest status.\n   * - Stops polling when the indexing process is completed or an error occurs.\n   * - Cleans up by resetting the indexing flags and clearing the `indexingInfos` property when the process is finalized.\n   */\n  trackIndexingProcess(token: string) {\n    // Clear the uploading flags\n    this.uploading$.next(false);\n    this.uploadingInfos$.next(undefined);\n    // Set the indexing flag to true\n    this.indexing$.next(true);\n\n    // Combine immediate API call with interval-based polling\n    this._subscription.add(\n      interval(this.pollingInterval).pipe(\n        startWith(0), // Trigger the API call immediately\n        switchMap(() => this.documentsUploadService.getIndexingStatus(token)),\n        tap((res: IndexingInfos) => this.indexingInfos$.next(res)),\n        takeWhile((response: IndexingInfos) => !response.isCompleted, true),\n        catchError((err) => {\n          console.error(err);\n          this.notificationsService.error(\"Error retrieving indexing status\");\n          return EMPTY;\n        }),\n        finalize(() => {\n          // Clear the indexing flags\n          this.indexing$.next(false);\n          this.indexingInfos$.next(undefined);\n        })\n      ).subscribe()\n    );\n  }\n\n  /**\n   * Calculates the indexing progress as a percentage of completed documents.\n   *\n   * @param infos - The indexing information containing the list of documents and their statuses.\n   * @returns The progress as a number between 0 and 1, where 0 indicates no progress and 1 indicates all documents are processed (processed means either indexed or errored).\n   *          Returns 0 if the input is invalid or there are no documents.\n   */\n  getIndexingProgress(infos: IndexingInfos | undefined): number {\n    if (!infos || !infos.docs) return 0;\n    const completed = infos.docs.filter(doc => doc.status === \"Indexed\" || doc.status === \"Error\").length;\n    const total = infos.docs.length;\n    return completed/total;\n  }\n\n}\n","<ng-container #flow=\"flow\" [flowConfig]=\"flowConfig\"></ng-container>\n<div class=\"file-upload-container\">\n  <input\n    type=\"file\"\n    flowButton\n    [flow]=\"flow.flowJs\"\n    multiple\n    hidden\n    #fileInput>\n  <div\n    flowDrop\n    [flow]=\"flow.flowJs\"\n    (dragenter)=\"onDragenter()\"\n    (dragleave)=\"onDragleave($event)\"\n    (drop)=\"onDrop()\"\n    (click)=\"fileInput.click()\"\n    class=\"dropzone\"\n    [ngClass]=\"{'dropzone--active': (dragging$ | async)}\"\n    [hidden]=\"(uploading$ | async) || (indexing$ | async)\">\n      <ng-container *ngIf=\"!(dragging$ | async); else draggingContent\">\n        <i class=\"fas fa-cloud-upload-alt\"></i>\n        <span>Drag and drop files here Or</span>\n        <span class=\"text-orange\">Click to browse</span>\n      </ng-container>\n      <ng-template #draggingContent>\n        <span>Drop files here</span>\n      </ng-template>\n  </div>\n\n  <div *ngIf=\"(uploading$ | async) || (indexing$ | async)\" class=\"dropzone dropzone--active\">\n    <ng-container *ngIf=\"(uploading$ | async); else indexingState\">\n      <i class=\"fas fa-spinner fa-pulse\"></i>\n      <span>\n        Uploading {{ (flow.transfers$ | async)!.transfers.length }} files\n      </span>\n      <span *ngIf=\"(uploadingInfos$ | async) as uploadingInfos\">{{ uploadingInfos.progress | number:'1.0-0' }}%</span>\n    </ng-container>\n    <ng-template #indexingState>\n      <i class=\"fas fa-spinner fa-pulse\"></i>\n      <ng-container *ngIf=\"indexingInfos$ | async as indexingInfo\">\n        <span>Indexing {{ indexingInfo.docs.length }} files</span>\n        <span>{{ getIndexingProgress(indexingInfo) * 100 | number:'1.0-0' }}%</span>\n      </ng-container>\n    </ng-template>\n  </div>\n\n  <ul *ngIf=\"errorAlerts.length > 0\" class=\"error-list mt-3\">\n    <li *ngFor=\"let error of errorAlerts\">\n      {{ error }}\n    </li>\n  </ul>\n\n  <div *ngIf=\"((flow.transfers$ | async)?.transfers?.length > 0) && !(uploading$ | async)\" class=\"file-list mt-3\">\n    <ul>\n      <li *ngFor=\"let transfer of (flow.transfers$ | async)!.transfers; trackBy: trackTransfer\">\n        <sq-format-icon [extension]=\"transfer.flowFile.getExtension()\" class=\"me-1\"></sq-format-icon>\n        <span [sqTooltip]=\"transfer.name\">{{ transfer.name }}</span>\n        <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [sqTooltip]=\"'Cancel'\"></i>\n      </li>\n    </ul>\n  </div>\n\n  <div class=\"d-flex mt-2\">\n    <button\n      type=\"button\"\n      class=\"btn btn-light cancel-btn me-2\"\n      (click)=\"flow.cancel()\"\n      [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n      Cancel\n    </button>\n    <button\n      type=\"button\"\n      class=\"upload-btn\"\n      (click)=\"upload()\"\n      [disabled]=\"!((flow.transfers$ | async)?.transfers?.length > 0) || (uploading$ | async) || (indexing$ | async)\">\n      Upload\n    </button>\n  </div>\n</div>\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG9jdW1lbnRzLXVwbG9hZC5tb2RlbC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Fzc2lzdGFudC9jaGF0L2RvY3VtZW50cy11cGxvYWQvZG9jdW1lbnRzLXVwbG9hZC5tb2RlbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiXG5leHBvcnQgdHlwZSBVcGxvYWRDb25maWcgPSB7XG4gIGRvY3VtZW50c1VwbG9hZEVuYWJsZWQ6IGJvb2xlYW47XG4gIGNvbnN0cmFpbnRzOiB7XG4gICAgbWF4VG90YWxGaWxlU2l6ZU1COiBudW1iZXI7XG4gICAgcmVtYWluaW5nRmlsZVNpemVNQjogbnVtYmVyO1xuICAgIG1heFRvdGFsRmlsZUNvdW50OiBudW1iZXI7XG4gICAgcmVtYWluaW5nRmlsZUNvdW50OiBudW1iZXI7XG4gICAgc3VwcG9ydGVkRmlsZUV4dGVuc2lvbnM6IHN0cmluZ1tdXG4gIH1cbn1cblxuZXhwb3J0IHR5cGUgVXBsb2FkUmVzcG9uc2UgPSB7XG4gIHN0YXR1c1Rva2VuOiBzdHJpbmc7XG4gIGV4ZWN1dGlvblRpbWU6IHN0cmluZztcbiAgY29ubmVjdG9yUmVzcG9uc2U/OiB7XG4gICAgY29sbGVjdGlvbjogc3RyaW5nO1xuICAgIGFjdGlvbjogc3RyaW5nO1xuICAgIHN0YXRzOiB7XG4gICAgICBuYkRvY1VwZGF0ZWQ6IG51bWJlcjtcbiAgICB9O1xuICB9O1xufVxuXG5leHBvcnQgdHlwZSBVcGxvYWRpbmdJbmZvcyA9IHtcbiAgbG9hZGVkOiBudW1iZXI7XG4gIHRvdGFsOiBudW1iZXI7XG4gIHByb2dyZXNzOiBudW1iZXI7XG59XG5cbmV4cG9ydCB0eXBlIFVwbG9hZFByb2dyZXNzRXZlbnQgPSB7IHR5cGU6ICd1cGxvYWRQcm9ncmVzcyc7IH0gJiBVcGxvYWRpbmdJbmZvcztcblxuZXhwb3J0IGludGVyZmFjZSBVcGxvYWRSZXNwb25zZUV2ZW50IHtcbiAgdHlwZTogJ3Jlc3BvbnNlJztcbiAgYm9keTogVXBsb2FkUmVzcG9uc2U7XG59XG5cbmV4cG9ydCB0eXBlIFVwbG9hZEV2ZW50ID0gVXBsb2FkUHJvZ3Jlc3NFdmVudCB8IFVwbG9hZFJlc3BvbnNlRXZlbnQ7XG5cbmV4cG9ydCB0eXBlIEluZGV4aW5nSW5mb3MgPSB7XG4gIGRvY3M6IHtcbiAgICBpZDogc3RyaW5nO1xuICAgIGZpbGVOYW1lOiBzdHJpbmc7XG4gICAgb3BlcmF0aW9uOiBcIkFkZFwiIHwgXCJVcGRhdGVcIjtcbiAgICBzdGF0dXM6IFwiSW5kZXhpbmdcIiB8IFwiSW5kZXhlZFwiIHwgXCJFcnJvclwiO1xuICAgIHByZXZpb3VzSW5kZXhhdGlvblRpbWU6IHN0cmluZyB8IG51bGw7XG4gICAgY3VycmVudEluZGV4YXRpb25UaW1lOiBzdHJpbmcgfCBudWxsO1xuICB9W107XG4gIGlzQ29tcGxldGVkOiBib29sZWFuO1xuICBleGVjdXRpb25UaW1lOiBzdHJpbmc7XG59O1xuXG5leHBvcnQgaW50ZXJmYWNlIFVwbG9hZGVkRG9jdW1lbnRzIHtcbiAgZG9jczogVXBsb2FkZWREb2N1bWVudFtdO1xuICBjb3VudDogbnVtYmVyO1xuICB0b3RhbENvdW50OiBudW1iZXI7XG4gIGV4ZWN1dGlvblRpbWU6IHN0cmluZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBVcGxvYWRlZERvY3VtZW50IHtcbiAgaWQ6IHN0cmluZztcbiAgZmlsZU5hbWU6IHN0cmluZztcbiAgdGl0bGU6IHN0cmluZztcbiAgZmlsZUV4dDogc3RyaW5nO1xuICBpbmRleGF0aW9uVGltZTogc3RyaW5nO1xuICBzaXplOiBzdHJpbmc7XG4gIHNpemVEaXNwbGF5OiBzdHJpbmc7XG4gIFtrZXk6IHN0cmluZ106IGFueTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBkZWxldGVEb2N1bWVudHNSZXNwb25zZSB7XG4gIGRlbGV0ZWRDb3VudDogbnVtYmVyO1xuICBleGVjdXRpb25UaW1lOiBzdHJpbmc7XG59XG4iXX0=