@sinequa/assistant 3.9.5 → 3.9.6-rc2

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 (69) hide show
  1. package/chat/chat-message/chat-message.component.d.ts +17 -25
  2. package/chat/chat-message/i18n/de.json +11 -0
  3. package/chat/chat-reference/chat-reference.component.d.ts +18 -8
  4. package/chat/chat-reference/i18n/de.json +4 -0
  5. package/chat/chat-settings-v3/i18n/de.json +14 -0
  6. package/chat/chat.component.d.ts +2 -1
  7. package/chat/debug-message/debug-message-details/debug-message-details.component.d.ts +2 -3
  8. package/chat/debug-message/i18n/de.json +3 -0
  9. package/chat/dialogs/i18n/de.json +19 -0
  10. package/chat/directives/copy-to-clipboard.directive.d.ts +8 -0
  11. package/chat/documents-upload/document-list/document-list.component.d.ts +4 -9
  12. package/chat/documents-upload/document-overview/document-overview.component.d.ts +3 -17
  13. package/chat/documents-upload/document-upload/document-upload.component.d.ts +3 -8
  14. package/chat/documents-upload/documents-upload.service.d.ts +7 -16
  15. package/chat/documents-upload/i18n/de.json +24 -0
  16. package/chat/i18n/de.json +42 -0
  17. package/chat/markdown-it-plugins/code-block.plugin.d.ts +2 -0
  18. package/chat/markdown-it-plugins/image-reference.plugin.d.ts +3 -0
  19. package/chat/markdown-it-plugins/link.plugin.d.ts +5 -0
  20. package/chat/markdown-it-plugins/page-reference.plugin.d.ts +3 -0
  21. package/chat/markdown-it-plugins/reference.plugin.d.ts +7 -0
  22. package/chat/public-api.d.ts +2 -1
  23. package/chat/references/i18n/de.json +6 -0
  24. package/chat/references/references.component.d.ts +43 -0
  25. package/chat/saved-chats/i18n/de.json +5 -0
  26. package/chat/services/app.service.d.ts +2 -1
  27. package/chat/services/custom-elements.service.d.ts +13 -0
  28. package/chat/smart-renderer/smart-renderer.d.ts +23 -0
  29. package/chat/token-progress-bar/i18n/de.json +4 -0
  30. package/chat/tooltip/tooltip.directive.d.ts +2 -2
  31. package/chat/types.d.ts +8 -42
  32. package/esm2022/chat/chat-message/chat-message.component.mjs +33 -135
  33. package/esm2022/chat/chat-reference/chat-reference.component.mjs +60 -26
  34. package/esm2022/chat/chat-settings-v3/chat-settings-v3.component.mjs +1 -1
  35. package/esm2022/chat/chat.component.mjs +48 -12
  36. package/esm2022/chat/chat.service.mjs +3 -2
  37. package/esm2022/chat/debug-message/debug-message-details/debug-message-details.component.mjs +2 -2
  38. package/esm2022/chat/debug-message/debug-message.service.mjs +4 -7
  39. package/esm2022/chat/directives/copy-to-clipboard.directive.mjs +68 -0
  40. package/esm2022/chat/documents-upload/document-list/document-list.component.mjs +21 -22
  41. package/esm2022/chat/documents-upload/document-overview/document-overview.component.mjs +8 -31
  42. package/esm2022/chat/documents-upload/document-upload/document-upload.component.mjs +12 -14
  43. package/esm2022/chat/documents-upload/documents-upload.service.mjs +25 -44
  44. package/esm2022/chat/markdown-it-plugins/code-block.plugin.mjs +14 -0
  45. package/esm2022/chat/markdown-it-plugins/image-reference.plugin.mjs +58 -0
  46. package/esm2022/chat/markdown-it-plugins/link.plugin.mjs +15 -0
  47. package/esm2022/chat/markdown-it-plugins/page-reference.plugin.mjs +60 -0
  48. package/esm2022/chat/markdown-it-plugins/reference.plugin.mjs +68 -0
  49. package/esm2022/chat/public-api.mjs +3 -2
  50. package/esm2022/chat/references/references.component.mjs +290 -0
  51. package/esm2022/chat/saved-chats/saved-chats.component.mjs +4 -3
  52. package/esm2022/chat/saved-chats/saved-chats.service.mjs +9 -12
  53. package/esm2022/chat/services/app.service.mjs +8 -2
  54. package/esm2022/chat/services/assistant-configuration.service.mjs +9 -18
  55. package/esm2022/chat/services/assistant-metadata.service.mjs +3 -3
  56. package/esm2022/chat/services/assistant-tokens-tracking.service.mjs +3 -6
  57. package/esm2022/chat/services/custom-elements.service.mjs +43 -0
  58. package/esm2022/chat/smart-renderer/smart-renderer.mjs +103 -0
  59. package/esm2022/chat/tooltip/tooltip.directive.mjs +9 -9
  60. package/esm2022/chat/types.mjs +2 -5
  61. package/fesm2022/sinequa-assistant-chat.mjs +908 -497
  62. package/fesm2022/sinequa-assistant-chat.mjs.map +1 -1
  63. package/package.json +4 -6
  64. package/chat/prompt.component.d.ts +0 -25
  65. package/chat/unified-plugins/embedded-image-reference.plugin.d.ts +0 -3
  66. package/chat/unified-plugins/embedded-page-reference.plugin.d.ts +0 -3
  67. package/esm2022/chat/prompt.component.mjs +0 -88
  68. package/esm2022/chat/unified-plugins/embedded-image-reference.plugin.mjs +0 -56
  69. package/esm2022/chat/unified-plugins/embedded-page-reference.plugin.mjs +0 -57
@@ -1,13 +1,13 @@
1
1
  import { CommonModule } from "@angular/common";
2
2
  import { Component, inject, Input, ViewChild, } from "@angular/core";
3
3
  import { NgxFlowModule } from "@flowjs/ngx-flow";
4
- import { BehaviorSubject, catchError, EMPTY, filter, finalize, interval, Subscription, switchMap, takeWhile, tap, startWith, of, } from "rxjs";
4
+ import { BehaviorSubject, catchError, EMPTY, finalize, interval, Subscription, switchMap, takeWhile, tap, startWith, of, take, } from "rxjs";
5
5
  import { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';
6
6
  import { isAuthenticated } from "@sinequa/atomic";
7
- import { InstanceManagerService } from "../../instance-manager.service";
8
7
  import { DocumentsUploadService } from "../documents-upload.service";
9
8
  import { FormatIconComponent } from '../../format-icon/format-icon.component';
10
9
  import { NotificationsService } from "../../services/notification.service";
10
+ import { AppService } from "../../services/app.service";
11
11
  import * as i0 from "@angular/core";
12
12
  import * as i1 from "@angular/common";
13
13
  import * as i2 from "@flowjs/ngx-flow";
@@ -34,13 +34,13 @@ export class DocumentUploadComponent {
34
34
  this.uploading$ = new BehaviorSubject(false);
35
35
  this.uploadingInfos$ = new BehaviorSubject(undefined);
36
36
  this._subscription = new Subscription();
37
- this.instanceManagerService = inject(InstanceManagerService);
38
37
  this.documentsUploadService = inject(DocumentsUploadService);
39
38
  this.notificationsService = inject(NotificationsService);
39
+ this.appService = inject(AppService);
40
40
  this.transloco = inject(TranslocoService);
41
41
  }
42
42
  ngOnInit() {
43
- this._subscription.add(of(isAuthenticated()).pipe(tap((_) => this.instantiateChatService()), switchMap((_) => this.chatService.assistantConfig$), filter((config) => !!config), tap((_) => this.documentsUploadService.init(this.chatService)), catchError((error) => {
43
+ this._subscription.add(of(isAuthenticated()).pipe(switchMap((_) => this.appService.init()), tap((_) => this.documentsUploadService.init()), catchError((error) => {
44
44
  console.error(error);
45
45
  this.notificationsService.error(error);
46
46
  return EMPTY;
@@ -63,9 +63,6 @@ export class DocumentUploadComponent {
63
63
  ngOnDestroy() {
64
64
  this._subscription.unsubscribe();
65
65
  }
66
- instantiateChatService() {
67
- this.chatService = this.instanceManagerService.getInstance(this.instanceId);
68
- }
69
66
  /**
70
67
  * Handles the submission of files.
71
68
  *
@@ -215,11 +212,14 @@ export class DocumentUploadComponent {
215
212
  this.indexing$.next(true);
216
213
  // Combine immediate API call with interval-based polling
217
214
  this._subscription.add(interval(this.pollingInterval).pipe(startWith(0), // Trigger the API call immediately
218
- switchMap(() => this.documentsUploadService.getIndexingStatus(token)), tap((res) => this.indexingInfos$.next(res)), takeWhile((response) => !response.isCompleted, true), catchError((err) => {
215
+ switchMap(() => this.documentsUploadService.getIndexingStatus(token)), tap((res) => this.indexingInfos$.next(res)), takeWhile((response) => !response.isCompleted, true), // Use takeWhileInclusive() pattern to ensure that the last value where isCompleted === true is also included.
216
+ catchError((err) => {
219
217
  console.error(err);
220
218
  this.notificationsService.error(this.transloco.translate('chatDocumentsUpload.errorIndexingStatus'));
221
219
  return EMPTY;
222
220
  }), finalize(() => {
221
+ // Refresh the list of uploaded documents at the end of indexing
222
+ this.documentsUploadService.getDocumentsList().pipe(take(1)).subscribe();
223
223
  // Clear the indexing flags
224
224
  this.indexing$.next(false);
225
225
  this.indexingInfos$.next(undefined);
@@ -240,17 +240,15 @@ export class DocumentUploadComponent {
240
240
  return completed / total;
241
241
  }
242
242
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DocumentUploadComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
243
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DocumentUploadComponent, isStandalone: true, selector: "sq-document-upload", inputs: { instanceId: "instanceId", pollingInterval: "pollingInterval" }, providers: [DocumentsUploadService, provideTranslocoScope('chat-documents-upload')], 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>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</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 {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\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>\n {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n </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 [title]=\"transfer.name\">{{ transfer.name }}</span>\n <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></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 {{ 'chatDocumentsUpload.cancel' | transloco }}\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 {{ 'chatDocumentsUpload.upload' | transloco }}\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:var(--ast-primary-bg, #f2f8fe)}.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:var(--ast-primary-bg, #f2f8fe);transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:var(--ast-secondary-bg, #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: "pipe", type: TranslocoPipe, name: "transloco" }] }); }
243
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DocumentUploadComponent, isStandalone: true, selector: "sq-document-upload", inputs: { pollingInterval: "pollingInterval" }, providers: [provideTranslocoScope('chat-documents-upload')], 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>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</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 {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\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>\n {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n </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 class=\"me-1\" [title]=\"transfer.name\">{{ transfer.name }}</span>\n <button\n type=\"button\"\n (click)=\"flow.cancelFile(transfer)\"\n [title]=\"'chatDocumentsUpload.cancel' | transloco\"\n class=\"bg-transparent border-0 p-0 m-0 cursor-pointer text-inherit leading-none inline-flex items-center justify-center\"\n >\n <i class=\"fas fa-trash\"></i>\n </button>\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 {{ 'chatDocumentsUpload.cancel' | transloco }}\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 {{ 'chatDocumentsUpload.upload' | transloco }}\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:var(--ast-primary-bg, #f2f8fe)}.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:var(--ast-primary-bg, #f2f8fe);transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:var(--ast-secondary-bg, #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: "pipe", type: TranslocoPipe, name: "transloco" }] }); }
244
244
  }
245
245
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DocumentUploadComponent, decorators: [{
246
246
  type: Component,
247
- args: [{ selector: "sq-document-upload", standalone: true, providers: [DocumentsUploadService, provideTranslocoScope('chat-documents-upload')], imports: [CommonModule, NgxFlowModule, FormatIconComponent, TranslocoPipe], 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>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</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 {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\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>\n {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n </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 [title]=\"transfer.name\">{{ transfer.name }}</span>\n <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></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 {{ 'chatDocumentsUpload.cancel' | transloco }}\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 {{ 'chatDocumentsUpload.upload' | transloco }}\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:var(--ast-primary-bg, #f2f8fe)}.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:var(--ast-primary-bg, #f2f8fe);transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:var(--ast-secondary-bg, #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"] }]
248
- }], propDecorators: { instanceId: [{
249
- type: Input
250
- }], pollingInterval: [{
247
+ args: [{ selector: "sq-document-upload", standalone: true, providers: [provideTranslocoScope('chat-documents-upload')], imports: [CommonModule, NgxFlowModule, FormatIconComponent, TranslocoPipe], 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>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n </ng-container>\n <ng-template #draggingContent>\n <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</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 {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\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>\n {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n </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 class=\"me-1\" [title]=\"transfer.name\">{{ transfer.name }}</span>\n <button\n type=\"button\"\n (click)=\"flow.cancelFile(transfer)\"\n [title]=\"'chatDocumentsUpload.cancel' | transloco\"\n class=\"bg-transparent border-0 p-0 m-0 cursor-pointer text-inherit leading-none inline-flex items-center justify-center\"\n >\n <i class=\"fas fa-trash\"></i>\n </button>\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 {{ 'chatDocumentsUpload.cancel' | transloco }}\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 {{ 'chatDocumentsUpload.upload' | transloco }}\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:var(--ast-primary-bg, #f2f8fe)}.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:var(--ast-primary-bg, #f2f8fe);transition:background-color .3s ease;color:var(--ast-primary-color, #005DA7)}.file-upload-container .dropzone--active{background-color:var(--ast-secondary-bg, #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"] }]
248
+ }], propDecorators: { pollingInterval: [{
251
249
  type: Input
252
250
  }], flow: [{
253
251
  type: ViewChild,
254
252
  args: ["flow", { static: true }]
255
253
  }] } });
256
- //# 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,EACT,EAAE,GACH,MAAM,MAAM,CAAC;AACd,OAAO,EAAE,qBAAqB,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAE5F,OAAO,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAElD,OAAO,EAAE,sBAAsB,EAAE,MAAM,gCAAgC,CAAC;AAExE,OAAO,EAAE,sBAAsB,EAAE,MAAM,6BAA6B,CAAC;AACrE,OAAO,EAAE,mBAAmB,EAAE,MAAM,yCAAyC,CAAC;AAG9E,OAAO,EAAE,oBAAoB,EAAE,MAAM,qCAAqC,CAAC;;;;AAU3E,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,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,yBAAoB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAC1C,cAAS,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;KA2PvD;IAzPC,QAAQ;QACN,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,EAAE,CAAC,eAAe,EAAE,CAAC,CAAC,IAAI,CACtB,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,CAAC;gBAChC,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAe,CAAC;gBAC9C,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;YACjC,CAAC;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,gBAAgB,EAAE,CAAC;gBACpC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,CAAC;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,CAAC;YACnG,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YACpC,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,8CAA8C,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC;QAClL,CAAC;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,CAAC;YACpL,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YACpC,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,4CAA4C,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,EAAE,CAAC,CAAC,CAAA;QAC/L,CAAC;QAED;;;;WAIG;aACE,CAAC;YACJ,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,uBAAuB,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC,EAAE,CAAC;oBAC7H,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;oBAClC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,8CAA8C,EAAE,EAAE,SAAS,EAAE,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC,CAAC;gBAC5H,CAAC;YACH,CAAC;QACH,CAAC;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,CAAC;YACD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC;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,CAAC;gBACpC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAuB,CAAC,CAAC;YACrD,CAAC;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;gBAC9B,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACpD,CAAC;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,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,yCAAyC,CAAC,CAAC,CAAC;YACrG,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,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,yCAAyC,CAAC,CAAC,CAAC;YACrG,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;+GA1RU,uBAAuB;mGAAvB,uBAAuB,2IAHvB,CAAC,sBAAsB,EAAE,qBAAqB,CAAC,uBAAuB,CAAC,CAAC,sIC1CrF,6wGAiFA,suHDtCY,YAAY,0cAAE,aAAa,yYAAE,mBAAmB,6EAAE,aAAa;;4FAE9D,uBAAuB;kBARnC,SAAS;+BACE,oBAAoB,cAGlB,IAAI,aACL,CAAC,sBAAsB,EAAE,qBAAqB,CAAC,uBAAuB,CAAC,CAAC,WAC1E,CAAC,YAAY,EAAE,aAAa,EAAE,mBAAmB,EAAE,aAAa,CAAC;8BAIjE,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  of,\n} from \"rxjs\";\nimport { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';\n\nimport { isAuthenticated } from \"@sinequa/atomic\";\n\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\";\nimport { NotificationsService } from \"../../services/notification.service\";\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, provideTranslocoScope('chat-documents-upload')],\n  imports: [CommonModule, NgxFlowModule, FormatIconComponent, TranslocoPipe]\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 instanceManagerService = inject(InstanceManagerService);\n  public documentsUploadService = inject(DocumentsUploadService);\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.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(this.transloco.translate('chatDocumentsUpload.filesNumberLimitExceeded', { max: this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileCount }));\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(this.transloco.translate('chatDocumentsUpload.filesSizeLimitExceeded', { max: Math.min(128, this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileSizeMB) }))\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(this.transloco.translate('chatDocumentsUpload.fileExtensionUnsupported', { extension: file.getExtension() }));\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(this.transloco.translate('chatDocumentsUpload.errorUploadingFiles'));\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 (pollingInterval: default = 1 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(this.transloco.translate('chatDocumentsUpload.errorIndexingStatus'));\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>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n        <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n      </ng-container>\n      <ng-template #draggingContent>\n        <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</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        {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\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>\n          {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n        </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 [title]=\"transfer.name\">{{ transfer.name }}</span>\n        <i class=\"fas fa-trash ms-1\" (click)=\"flow.cancelFile(transfer)\" [title]=\"'chatDocumentsUpload.cancel' | transloco\"></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      {{ 'chatDocumentsUpload.cancel' | transloco }}\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      {{ 'chatDocumentsUpload.upload' | transloco }}\n    </button>\n  </div>\n</div>\n"]}
254
+ //# 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,QAAQ,EACR,QAAQ,EACR,YAAY,EACZ,SAAS,EACT,SAAS,EACT,GAAG,EACH,SAAS,EACT,EAAE,EACF,IAAI,GACL,MAAM,MAAM,CAAC;AACd,OAAO,EAAE,qBAAqB,EAAE,aAAa,EAAE,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAE5F,OAAO,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAElD,OAAO,EAAE,sBAAsB,EAAE,MAAM,6BAA6B,CAAC;AACrE,OAAO,EAAE,mBAAmB,EAAE,MAAM,yCAAyC,CAAC;AAE9E,OAAO,EAAE,oBAAoB,EAAE,MAAM,qCAAqC,CAAC;AAC3E,OAAO,EAAE,UAAU,EAAE,MAAM,4BAA4B,CAAC;;;;AAUxD,MAAM,OAAO,uBAAuB;IARpC;QASE,+FAA+F;QACtF,oBAAe,GAAY,IAAI,CAAC;QAIhC,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,2BAAsB,GAAG,MAAM,CAAC,sBAAsB,CAAC,CAAC;QACxD,yBAAoB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QACpD,eAAU,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC;QACtB,cAAS,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;KAuPvD;IArPC,QAAQ;QACN,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,EAAE,CAAC,eAAe,EAAE,CAAC,CAAC,IAAI,CACtB,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC,EACxC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,sBAAsB,CAAC,IAAI,EAAE,CAAC,EAC9C,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,CAAC;gBAChC,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAe,CAAC;gBAC9C,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,CAAC;YACjC,CAAC;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,gBAAgB,EAAE,CAAC;gBACpC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,CAAC;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;;;;;;;;;;;;;;;;;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,CAAC;YACnG,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YACpC,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,8CAA8C,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,kBAAkB,EAAE,CAAC,CAAC,CAAC;QAClL,CAAC;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,CAAC;YACpL,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YACpC,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,4CAA4C,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,mBAAmB,CAAC,EAAE,CAAC,CAAC,CAAA;QAC/L,CAAC;QAED;;;;WAIG;aACE,CAAC;YACJ,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,aAAa,CAAC,KAAM,CAAC,WAAW,CAAC,uBAAuB,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC,EAAE,CAAC;oBAC7H,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;oBAClC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,8CAA8C,EAAE,EAAE,SAAS,EAAE,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC,CAAC;gBAC5H,CAAC;YACH,CAAC;QACH,CAAC;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,CAAC;YACD,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC7B,CAAC;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,CAAC;gBACpC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,KAAuB,CAAC,CAAC;YACrD,CAAC;YACD,IAAI,KAAK,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;gBAC9B,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACpD,CAAC;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,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,yCAAyC,CAAC,CAAC,CAAC;YACrG,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,EAAE,8GAA8G;QACnL,UAAU,CAAC,CAAC,GAAG,EAAE,EAAE;YACjB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YACnB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,yCAAyC,CAAC,CAAC,CAAC;YACrG,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,EACF,QAAQ,CAAC,GAAG,EAAE;YACZ,gEAAgE;YAChE,IAAI,CAAC,sBAAsB,CAAC,gBAAgB,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,CAAC;YACzE,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;+GAnRU,uBAAuB;mGAAvB,uBAAuB,iHAHvB,CAAC,qBAAqB,CAAC,uBAAuB,CAAC,CAAC,sICxC7D,2/GAwFA,suHD/CY,YAAY,0cAAE,aAAa,yYAAE,mBAAmB,6EAAE,aAAa;;4FAE9D,uBAAuB;kBARnC,SAAS;+BACE,oBAAoB,cAGlB,IAAI,aACL,CAAC,qBAAqB,CAAC,uBAAuB,CAAC,CAAC,WAClD,CAAC,YAAY,EAAE,aAAa,EAAE,mBAAmB,EAAE,aAAa,CAAC;8BAIjE,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  finalize,\n  interval,\n  Subscription,\n  switchMap,\n  takeWhile,\n  tap,\n  startWith,\n  of,\n  take,\n} from \"rxjs\";\nimport { provideTranslocoScope, TranslocoPipe, TranslocoService } from '@jsverse/transloco';\n\nimport { isAuthenticated } from \"@sinequa/atomic\";\n\nimport { DocumentsUploadService } from \"../documents-upload.service\";\nimport { FormatIconComponent } from '../../format-icon/format-icon.component';\nimport { IndexingInfos, UploadEvent, UploadingInfos } from \"../documents-upload.model\";\nimport { NotificationsService } from \"../../services/notification.service\";\nimport { AppService } from \"../../services/app.service\";\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: [provideTranslocoScope('chat-documents-upload')],\n  imports: [CommonModule, NgxFlowModule, FormatIconComponent, TranslocoPipe]\n})\nexport class DocumentUploadComponent implements OnInit, OnDestroy {\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  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 documentsUploadService = inject(DocumentsUploadService);\n  public notificationsService = inject(NotificationsService);\n  public appService = inject(AppService);\n  private readonly transloco = inject(TranslocoService);\n\n  ngOnInit(): void {\n    this._subscription.add(\n      of(isAuthenticated()).pipe(\n          switchMap((_) => this.appService.init()),\n          tap((_) => this.documentsUploadService.init()),\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  /**\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(this.transloco.translate('chatDocumentsUpload.filesNumberLimitExceeded', { max: this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileCount }));\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(this.transloco.translate('chatDocumentsUpload.filesSizeLimitExceeded', { max: Math.min(128, this.documentsUploadService.uploadConfig$.value!.constraints.remainingFileSizeMB) }))\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(this.transloco.translate('chatDocumentsUpload.fileExtensionUnsupported', { extension: file.getExtension() }));\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(this.transloco.translate('chatDocumentsUpload.errorUploadingFiles'));\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 (pollingInterval: default = 1 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), // Use takeWhileInclusive() pattern to ensure that the last value where isCompleted === true is also included.\n        catchError((err) => {\n          console.error(err);\n          this.notificationsService.error(this.transloco.translate('chatDocumentsUpload.errorIndexingStatus'));\n          return EMPTY;\n        }),\n        finalize(() => {\n          // Refresh the list of uploaded documents at the end of indexing\n          this.documentsUploadService.getDocumentsList().pipe(take(1)).subscribe();\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>{{ 'chatDocumentsUpload.dragAndDrop' | transloco }}</span>\n        <span class=\"text-orange\">{{ 'chatDocumentsUpload.clickToBrowse' | transloco }}</span>\n      </ng-container>\n      <ng-template #draggingContent>\n        <span>{{ 'chatDocumentsUpload.dropFiles' | transloco }}</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        {{ 'chatDocumentsUpload.uploadingFiles' | transloco: { count: (flow.transfers$ | async)!.transfers.length } }}\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>\n          {{ 'chatDocumentsUpload.indexingFiles' | transloco: { count: indexingInfo.docs.length } }}\n        </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 class=\"me-1\" [title]=\"transfer.name\">{{ transfer.name }}</span>\n        <button\n          type=\"button\"\n          (click)=\"flow.cancelFile(transfer)\"\n          [title]=\"'chatDocumentsUpload.cancel' | transloco\"\n          class=\"bg-transparent border-0 p-0 m-0 cursor-pointer text-inherit leading-none inline-flex items-center justify-center\"\n        >\n          <i class=\"fas fa-trash\"></i>\n    </button>\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      {{ 'chatDocumentsUpload.cancel' | transloco }}\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      {{ 'chatDocumentsUpload.upload' | transloco }}\n    </button>\n  </div>\n</div>\n"]}
@@ -1,46 +1,30 @@
1
1
  import { Injectable, inject } from '@angular/core';
2
- import { BehaviorSubject, catchError, filter, from, map, take } from 'rxjs';
2
+ import { BehaviorSubject, catchError, filter, from, map, take, tap } from 'rxjs';
3
3
  import { HttpClient, HttpEventType, HttpHeaders } from '@angular/common/http';
4
4
  import { AppService } from '../services/app.service';
5
5
  import { getToken, globalConfig, post } from '@sinequa/atomic';
6
6
  import * as i0 from "@angular/core";
7
7
  export class DocumentsUploadService {
8
8
  constructor() {
9
- /** Documents' upload configuration. */
9
+ this.REQUEST_URL = "api/v1/plugin/SinequaAssistantREST";
10
+ /** Documents upload configuration. */
10
11
  this.uploadConfig$ = new BehaviorSubject(undefined);
12
+ /** Emit the list of indexed documents */
13
+ this.uploadedDocuments$ = new BehaviorSubject([]);
11
14
  this.appService = inject(AppService);
12
15
  this.http = inject(HttpClient);
13
16
  }
14
17
  /**
15
- * Initializes the file upload service with the provided chat service instance.
16
- *
17
- * @param chatService - An instance of ChatService. If not provided, an error is thrown.
18
- * @throws {Error} If the chatService instance is not provided or if there is an error during initialization.
18
+ * Initializes the file upload service.
19
19
  */
20
- init(chatService) {
21
- if (!chatService)
22
- throw new Error('A chatService instance must be provided');
23
- this.chatService = chatService;
20
+ init() {
24
21
  try {
25
- this.getRequestsUrl();
26
22
  this.getUploadConfig();
27
23
  }
28
24
  catch (error) {
29
25
  throw new Error(error);
30
26
  }
31
27
  }
32
- /**
33
- * Fetch the endpoint to use for the file upload related requests
34
- * @throws {Error} If the property 'restEndpoint' is not provided in the assistant configuration
35
- */
36
- getRequestsUrl() {
37
- if (!!this.chatService.assistantConfig$.value.connectionSettings.restEndpoint) {
38
- this.REQUEST_URL = this.chatService.assistantConfig$.value.connectionSettings.restEndpoint;
39
- }
40
- else {
41
- throw new Error(`The property 'restEndpoint' must be provided when attempting to use 'REST' in assistant instance`);
42
- }
43
- }
44
28
  /**
45
29
  * Retrieves the upload configuration for documents.
46
30
  * The response, which contains the upload configuration, is emitted to the `uploadConfig$` observable.
@@ -50,10 +34,9 @@ export class DocumentsUploadService {
50
34
  getUploadConfig() {
51
35
  const data = {
52
36
  action: "documentsUploadConfigGet",
53
- appName: this.appService.appName,
54
- debug: this.chatService.assistantConfig$.value.defaultValues.debug
37
+ appName: this.appService.appName
55
38
  };
56
- from(post(`plugin/${this.REQUEST_URL}`, data)).pipe(take(1), catchError((error) => {
39
+ from(post(this.REQUEST_URL, data)).pipe(take(1), catchError((error) => {
57
40
  throw new Error(`Error invoking documentsUploadConfigGet: ${error}`);
58
41
  })).subscribe((res) => this.uploadConfig$.next(res));
59
42
  }
@@ -118,8 +101,7 @@ export class DocumentsUploadService {
118
101
  // Add the required "data" field as a JSON string
119
102
  const data = {
120
103
  action: "documentsupload",
121
- appName: this.appService.appName,
122
- debug: this.chatService.assistantConfig$.value.defaultValues.debug
104
+ appName: this.appService.appName
123
105
  };
124
106
  formData.append('data', JSON.stringify(data));
125
107
  let headers = new HttpHeaders({ 'Accept': 'application/json' });
@@ -133,7 +115,7 @@ export class DocumentsUploadService {
133
115
  };
134
116
  if (enableProgress)
135
117
  options['reportProgress'] = true;
136
- const url = `${globalConfig.backendUrl}/${globalConfig.apiPath}/plugin/${this.REQUEST_URL}`;
118
+ const url = `${globalConfig.backendUrl}/${this.REQUEST_URL}`;
137
119
  /**
138
120
  * Need to use the HttpClient directly to avoid the JSONMethodPluginService's behavior
139
121
  * because it is altering the request and thus making it unable to properly set the Content-Type header with the boundary
@@ -174,10 +156,9 @@ export class DocumentsUploadService {
174
156
  const data = {
175
157
  action: "documentsUploadStatus",
176
158
  appName: this.appService.appName,
177
- statusToken: token,
178
- debug: this.chatService.assistantConfig$.value.defaultValues.debug
159
+ statusToken: token
179
160
  };
180
- return from(post(`plugin/${this.REQUEST_URL}`, data)).pipe(catchError((error) => {
161
+ return from(post(this.REQUEST_URL, data)).pipe(catchError((error) => {
181
162
  throw new Error(`Error invoking documentsUploadStatus: ${error}`);
182
163
  }));
183
164
  }
@@ -206,8 +187,7 @@ export class DocumentsUploadService {
206
187
  getDocumentsList(columns, skip, count) {
207
188
  const data = {
208
189
  action: "documentsList",
209
- appName: this.appService.appName,
210
- debug: this.chatService.assistantConfig$.value.defaultValues.debug
190
+ appName: this.appService.appName
211
191
  };
212
192
  if (columns)
213
193
  data["columns"] = columns;
@@ -215,7 +195,7 @@ export class DocumentsUploadService {
215
195
  data["skip"] = skip;
216
196
  if (count)
217
197
  data["count"] = count;
218
- return from(post(`plugin/${this.REQUEST_URL}`, data)).pipe(catchError((error) => {
198
+ return from(post(this.REQUEST_URL, data)).pipe(tap((res) => this.uploadedDocuments$.next(res.docs)), catchError((error) => {
219
199
  throw new Error(`Error invoking documentsList: ${error}`);
220
200
  }));
221
201
  }
@@ -233,10 +213,9 @@ export class DocumentsUploadService {
233
213
  const data = {
234
214
  action: "documentsDelete",
235
215
  docIds,
236
- appName: this.appService.appName,
237
- debug: this.chatService.assistantConfig$.value.defaultValues.debug
216
+ appName: this.appService.appName
238
217
  };
239
- return from(post(`plugin/${this.REQUEST_URL}`, data)).pipe(catchError((error) => {
218
+ return from(post(this.REQUEST_URL, data)).pipe(catchError((error) => {
240
219
  throw new Error(`Error invoking documentsDelete: ${error}`);
241
220
  }));
242
221
  }
@@ -252,10 +231,9 @@ export class DocumentsUploadService {
252
231
  deleteAllDocuments() {
253
232
  const data = {
254
233
  action: "documentsDeleteAll",
255
- appName: this.appService.appName,
256
- debug: this.chatService.assistantConfig$.value.defaultValues.debug
234
+ appName: this.appService.appName
257
235
  };
258
- return from(post(`plugin/${this.REQUEST_URL}`, data)).pipe(catchError((error) => {
236
+ return from(post(this.REQUEST_URL, data)).pipe(catchError((error) => {
259
237
  throw new Error(`Error invoking documentsDeleteAll: ${error}`);
260
238
  }));
261
239
  }
@@ -283,9 +261,12 @@ export class DocumentsUploadService {
283
261
  return bytes / (1024 * 1024);
284
262
  }
285
263
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DocumentsUploadService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
286
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DocumentsUploadService }); }
264
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DocumentsUploadService, providedIn: 'root' }); }
287
265
  }
288
266
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DocumentsUploadService, decorators: [{
289
- type: Injectable
267
+ type: Injectable,
268
+ args: [{
269
+ providedIn: 'root'
270
+ }]
290
271
  }] });
291
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"documents-upload.service.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/documents-upload/documents-upload.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { ChatService } from '../chat.service';\nimport { BehaviorSubject, catchError, filter, from, map, Observable, take } from 'rxjs';\nimport { deleteDocumentsResponse, IndexingInfos, UploadConfig, UploadedDocuments, UploadEvent } from './documents-upload.model';\nimport { HttpClient, HttpEventType, HttpHeaders, HttpResponse, HttpUploadProgressEvent } from '@angular/common/http';\nimport { AppService } from '../services/app.service';\nimport { getToken, globalConfig, post } from '@sinequa/atomic';\n\n@Injectable()\nexport class DocumentsUploadService {\n\n  /** Name of the assistant plugin. */\n  public REQUEST_URL: string;\n  /** Documents' upload configuration. */\n  uploadConfig$ = new BehaviorSubject<UploadConfig | undefined>(undefined);\n  public chatService: ChatService;\n\n  public appService = inject(AppService);\n  public http = inject(HttpClient);\n\n  /**\n   * Initializes the file upload service with the provided chat service instance.\n   *\n   * @param chatService - An instance of ChatService. If not provided, an error is thrown.\n   * @throws {Error} If the chatService instance is not provided or if there is an error during initialization.\n   */\n  init(chatService: ChatService | undefined): void {\n    if (!chatService) throw new Error('A chatService instance must be provided');\n\n    this.chatService = chatService;\n    try {\n      this.getRequestsUrl();\n      this.getUploadConfig();\n    } catch (error) {\n      throw new Error(error);\n    }\n  }\n\n  /**\n   * Fetch the endpoint to use for the file upload related requests\n   * @throws {Error} If the property 'restEndpoint' is not provided in the assistant configuration\n   */\n  getRequestsUrl() {\n    if (!!this.chatService.assistantConfig$.value!.connectionSettings.restEndpoint) {\n      this.REQUEST_URL = this.chatService.assistantConfig$.value!.connectionSettings.restEndpoint;\n    } else {\n      throw new Error(`The property 'restEndpoint' must be provided when attempting to use 'REST' in assistant instance`);\n    }\n  }\n\n  /**\n   * Retrieves the upload configuration for documents.\n   * The response, which contains the upload configuration, is emitted to the `uploadConfig$` observable.\n   *\n   * @throws {Error} If there is an error invoking the `documentsUploadConfigGet` action.\n   */\n  public getUploadConfig(): void {\n    const data = {\n      action: \"documentsUploadConfigGet\",\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    from(post<UploadConfig>(`plugin/${this.REQUEST_URL}`, data)).pipe(\n      take(1),\n      catchError((error) => {\n        throw new Error(`Error invoking documentsUploadConfigGet: ${error}`);\n      })\n    ).subscribe((res: UploadConfig) => this.uploadConfig$.next(res));\n  }\n\n  /**\n   * Uploads documents to the server.\n   *\n   * This method takes a FormData object which should contain all files to be uploaded.\n   * Each file should be appended to the FormData object with its original name.\n   *\n   * For example:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * ```\n   *\n   * @param formData - The FormData object containing the files to be uploaded.\n   * - Each file should be appended to the FormData object with its original name.\n   * For example:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * ```\n   *\n   * @param enableProgress - (Optional) A boolean parameter that controls whether progress reporting is enabled.\n   * - Defaults to `true`.\n   * - If `true`, the `reportProgress` option is set in the request options, enabling the track of upload progress based on the underlying browser's progress events.\n   * - If `false`, progress events are not reported.\n   *\n   * @returns An Observable that emits events of type `UploadEvent`:\n   * - `UploadProgressEvent`: Emitted during the upload process, containing:\n   *   - *  `type`: \"uploadProgress\" - The type of the event.\n   *   - *  `loaded`: number - The number of bytes uploaded so far.\n   *   - *  `total`: number - The total number of bytes to be uploaded (if available).\n   *   - *  `progress`: number - The percentage of upload progress (calculated as `(loaded / total) * 100`).\n   * - `UploadResponseEvent`: Emitted when the upload is complete, containing:\n   *   - *  `type`: \"response\" - The type of the event.\n   *   - *  `body`: `UploadResponse` - The server's response, including:\n   *     - * - * `statusToken`: string - A token to track the status of the indexing process.\n   *     - * - *  `executionTime`: string - The time taken to execute the upload.\n   *     - * - *  `connectorResponse` (optional): Additional response data from the connector, containing:\n   *       - * - * - *  `collection`: string - The collection name.\n   *       - * - * - *  `action`: string - The action performed.\n   *       - * - * - *  `stats`: object - Statistics about the upload, including:\n   *         - * - * - * - * `nbDocUpdated`: number - The number of documents updated.\n   *\n   * @throws {Error} If there is an error invoking the `documentsupload` action.\n   *\n   * Example usage:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * this.documentsUploadService.uploadDocuments(formData).subscribe(event => {\n   *   if (event.type === 'uploadProgress') {\n   *     console.log(`Progress: ${event.progress}%`);\n   *   } else if (event.type === 'response') {\n   *     console.log('Upload complete:', event.body);\n   *   }\n   * });\n   * ```\n   */\n  public uploadDocuments(formData: FormData, enableProgress: boolean = true): Observable<UploadEvent> {\n    // Add the required \"data\" field as a JSON string\n    const data = {\n      action: \"documentsupload\",\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n    formData.append('data', JSON.stringify(data));\n\n    let headers = new HttpHeaders({ 'Accept': 'application/json' });\n    const token = getToken();\n    if(token){\n      headers = headers.append('sinequa-csrf-token',token);\n    }\n    const options: any  = {\n      headers: headers,\n      observe: 'events',\n    };\n    if (enableProgress) options['reportProgress'] = true;\n\n    const url = `${globalConfig.backendUrl}/${globalConfig.apiPath}/plugin/${this.REQUEST_URL}`;\n    /**\n     * Need to use the HttpClient directly to avoid the JSONMethodPluginService's behavior\n     * because it is altering the request and thus making it unable to properly set the Content-Type header with the boundary\n     */\n    return this.http.post<any>(url, formData, options).pipe(\n      filter((event): event is HttpUploadProgressEvent | HttpResponse<any> =>\n        event.type === HttpEventType.Response || (enableProgress && event.type === HttpEventType.UploadProgress)\n      ),\n      map((event): UploadEvent => {\n        switch (event.type) {\n          case HttpEventType.UploadProgress: {\n            const { loaded = 0, total = 0 } = event;\n            return { type: 'uploadProgress', loaded, total, progress: total > 0 ? Math.round((loaded / total) * 100) : 0 };\n          }\n          case HttpEventType.Response:\n            return { type: 'response', body: event.body };\n          default:\n            throw new Error(\"Unexpected event type\"); // Should not reach here, but for typescript sake.\n        }\n      }),\n      catchError((error) => {\n        throw new Error(`Error invoking documentsupload: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Retrieves the indexing status of a predefined uploaded document(s)\n   * based on the provided status token returned from the upload process.\n   *\n   * @param token - The status token used to track the indexing process.\n   * @returns An Observable that emits the server response of type `IndexingResponse`.\n   * The `IndexingResponse` object contains:\n   * - `docs`: array - An array of document status objects, each containing:\n   *   - * `id`: string - The document ID.\n   *   - * `fileName`: string - The name of the file.\n   *   - * `operation`: \"Add\" | \"Update\" - The operation performed on the document.\n   *   - * `status`: \"Indexing\" | \"Indexed\" | \"Error\" - The status of the document.\n   *   - * `previousIndexationTime`: string | null - The previous indexation time.\n   *   - * `currentIndexationTime`: string | null - The current indexation time.\n   * - `executionTime`: string - The time taken to retrieve the indexing status.\n   * @throws {Error} If there is an error invoking the `documentsUploadStatus` action.\n   */\n  public getIndexingStatus(token: string): Observable<IndexingInfos> {\n    const data = {\n      action: \"documentsUploadStatus\",\n      appName: this.appService.appName,\n      statusToken: token,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    return from(post<IndexingInfos>(`plugin/${this.REQUEST_URL}`, data)).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsUploadStatus: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Retrieves a list of uploaded documents.\n   *\n   * @param {string[]} [columns] - Optional array of additional index columns to include for each document.\n   * @param {number} [skip] - Optional number of documents to skip for pagination. If set, count also needs to be set. No skip if not set.\n   * @param {number} [count] - Optional number of documents to take for pagination (unlimited if not set).\n   * @returns {Observable<UploadedDocuments>} An observable that emits the list of uploaded documents.\n   * The `UploadedDocuments` object contains:\n   * - `docs`: array - An array of objects representing a single document with the following properties:\n   *   - * `id`: string - A unique identifier for the document.\n   *   - * `fileName`: string - The name of the file.\n   *   - * `title`: string - The title of the document.\n   *   - * `fileExt`: string - The file extension of the document.\n   *   - * `indexationTime`: string - The time when the document was indexed.\n   *   - * `size`: number - The size of the document.\n   *   - * `sizeDisplay`: string - A human-readable representation of the document size.\n   *   - * `[key: string]`: any - Additional columns asked in the request, value of the column for the document.\n   * - `count`: number - The count of documents in the current batch or subset.\n   * - `totalCount`: number - The total count of documents available.\n   * - `executionTime`: string - The time taken to process the documents.\n   * @throws {Error} Throws an error if the request fails.\n   */\n  public getDocumentsList(columns?: string[], skip?: number, count?: number): Observable<UploadedDocuments> {\n    const data = {\n      action: \"documentsList\",\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    if (columns) data[\"columns\"] = columns;\n    if (skip) data[\"skip\"] = skip;\n    if (count) data[\"count\"] = count;\n\n    return from(post<UploadedDocuments>(`plugin/${this.REQUEST_URL}`, data)).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsList: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Deletes the uploaded documents with the specified IDs.\n   *\n   * @param docIds - An array of document IDs to delete.\n   * @returns An observable that emits the server response of type `deleteDocumentsResponse`.\n   * The `deleteDocumentsResponse` object contains:\n   * - `deletedCount`: number - The number of deleted documents.\n   * - `executionTime`: string - The time taken to delete the documents.\n   * @throws {Error} If there is an error invoking the `documentsDelete` action.\n   */\n  public deleteDocuments(docIds: string[]): Observable<deleteDocumentsResponse> {\n    const data = {\n      action: \"documentsDelete\",\n      docIds,\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    return from(post<deleteDocumentsResponse>(`plugin/${this.REQUEST_URL}`, data)).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsDelete: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Deletes all uploaded documents.\n   *\n   * @returns An observable that emits the server response of type `deleteDocumentsResponse`.\n   * The `deleteDocumentsResponse` object contains:\n   * - `deletedCount`: number - The number of deleted documents.\n   * - `executionTime`: string - The time taken to delete the documents.\n   * @throws {Error} If there is an error invoking the `documentsDeleteAll` action.\n   */\n  public deleteAllDocuments(): Observable<deleteDocumentsResponse> {\n    const data = {\n      action: \"documentsDeleteAll\",\n      appName: this.appService.appName,\n      debug: this.chatService.assistantConfig$.value!.defaultValues.debug\n    };\n\n    return from(post<deleteDocumentsResponse>(`plugin/${this.REQUEST_URL}`, data)).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsDeleteAll: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Formats the given file size in bytes into a human-readable string.\n   *\n   * @param bytes - The size of the file in bytes.\n   * @returns A string representing the file size in a human-readable format (e.g., \"10.24 KB\", \"1.00 MB\").\n   */\n  public formatFileSize(bytes: number): string {\n    if (bytes === 0) return \"0 Bytes\";\n    const k = 1024;\n    const sizes = [\"Bytes\", \"KB\", \"MB\", \"GB\", \"TB\"];\n    const i = Math.floor(Math.log(bytes) / Math.log(k));\n    return (\n      Number.parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + \" \" + sizes[i]\n    );\n  }\n\n  /**\n   * Converts the given file size in bytes into megabytes.\n   *\n   * @param bytes - The size of the file in bytes.\n   * @returns The size of the file in megabytes.\n   */\n  public convertBytesToMB(bytes: number): number {\n    return bytes / (1024 * 1024);\n  }\n\n}\n"]}
272
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"documents-upload.service.js","sourceRoot":"","sources":["../../../../../projects/assistant/chat/documents-upload/documents-upload.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { BehaviorSubject, catchError, filter, from, map, Observable, take, tap } from 'rxjs';\nimport { deleteDocumentsResponse, IndexingInfos, UploadConfig, UploadedDocument, UploadedDocuments, UploadEvent } from './documents-upload.model';\nimport { HttpClient, HttpEventType, HttpHeaders, HttpResponse, HttpUploadProgressEvent } from '@angular/common/http';\nimport { AppService } from '../services/app.service';\nimport { getToken, globalConfig, post } from '@sinequa/atomic';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class DocumentsUploadService {\n\n  public readonly REQUEST_URL = \"api/v1/plugin/SinequaAssistantREST\";\n  /** Documents upload configuration. */\n  public uploadConfig$ = new BehaviorSubject<UploadConfig | undefined>(undefined);\n\n  /** Emit the list of indexed documents */\n  public uploadedDocuments$ = new BehaviorSubject<UploadedDocument[] | undefined>([]);\n\n  public appService = inject(AppService);\n  public http = inject(HttpClient);\n\n  /**\n   * Initializes the file upload service.\n   */\n  init(): void {\n    try {\n      this.getUploadConfig();\n    } catch (error) {\n      throw new Error(error);\n    }\n  }\n\n  /**\n   * Retrieves the upload configuration for documents.\n   * The response, which contains the upload configuration, is emitted to the `uploadConfig$` observable.\n   *\n   * @throws {Error} If there is an error invoking the `documentsUploadConfigGet` action.\n   */\n  public getUploadConfig(): void {\n    const data = {\n      action: \"documentsUploadConfigGet\",\n      appName: this.appService.appName\n    };\n\n    from(post<UploadConfig>(this.REQUEST_URL, data)).pipe(\n      take(1),\n      catchError((error) => {\n        throw new Error(`Error invoking documentsUploadConfigGet: ${error}`);\n      })\n    ).subscribe((res: UploadConfig) => this.uploadConfig$.next(res));\n  }\n\n  /**\n   * Uploads documents to the server.\n   *\n   * This method takes a FormData object which should contain all files to be uploaded.\n   * Each file should be appended to the FormData object with its original name.\n   *\n   * For example:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * ```\n   *\n   * @param formData - The FormData object containing the files to be uploaded.\n   * - Each file should be appended to the FormData object with its original name.\n   * For example:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * ```\n   *\n   * @param enableProgress - (Optional) A boolean parameter that controls whether progress reporting is enabled.\n   * - Defaults to `true`.\n   * - If `true`, the `reportProgress` option is set in the request options, enabling the track of upload progress based on the underlying browser's progress events.\n   * - If `false`, progress events are not reported.\n   *\n   * @returns An Observable that emits events of type `UploadEvent`:\n   * - `UploadProgressEvent`: Emitted during the upload process, containing:\n   *   - *  `type`: \"uploadProgress\" - The type of the event.\n   *   - *  `loaded`: number - The number of bytes uploaded so far.\n   *   - *  `total`: number - The total number of bytes to be uploaded (if available).\n   *   - *  `progress`: number - The percentage of upload progress (calculated as `(loaded / total) * 100`).\n   * - `UploadResponseEvent`: Emitted when the upload is complete, containing:\n   *   - *  `type`: \"response\" - The type of the event.\n   *   - *  `body`: `UploadResponse` - The server's response, including:\n   *     - * - * `statusToken`: string - A token to track the status of the indexing process.\n   *     - * - *  `executionTime`: string - The time taken to execute the upload.\n   *     - * - *  `connectorResponse` (optional): Additional response data from the connector, containing:\n   *       - * - * - *  `collection`: string - The collection name.\n   *       - * - * - *  `action`: string - The action performed.\n   *       - * - * - *  `stats`: object - Statistics about the upload, including:\n   *         - * - * - * - * `nbDocUpdated`: number - The number of documents updated.\n   *\n   * @throws {Error} If there is an error invoking the `documentsupload` action.\n   *\n   * Example usage:\n   * ```typescript\n   * const formData = new FormData();\n   * formData.append(file.name, file.file);\n   * this.documentsUploadService.uploadDocuments(formData).subscribe(event => {\n   *   if (event.type === 'uploadProgress') {\n   *     console.log(`Progress: ${event.progress}%`);\n   *   } else if (event.type === 'response') {\n   *     console.log('Upload complete:', event.body);\n   *   }\n   * });\n   * ```\n   */\n  public uploadDocuments(formData: FormData, enableProgress: boolean = true): Observable<UploadEvent> {\n    // Add the required \"data\" field as a JSON string\n    const data = {\n      action: \"documentsupload\",\n      appName: this.appService.appName\n    };\n    formData.append('data', JSON.stringify(data));\n\n    let headers = new HttpHeaders({ 'Accept': 'application/json' });\n    const token = getToken();\n    if(token){\n      headers = headers.append('sinequa-csrf-token',token);\n    }\n    const options: any  = {\n      headers: headers,\n      observe: 'events',\n    };\n    if (enableProgress) options['reportProgress'] = true;\n\n    const url = `${globalConfig.backendUrl}/${this.REQUEST_URL}`;\n    /**\n     * Need to use the HttpClient directly to avoid the JSONMethodPluginService's behavior\n     * because it is altering the request and thus making it unable to properly set the Content-Type header with the boundary\n     */\n    return this.http.post<any>(url, formData, options).pipe(\n      filter((event): event is HttpUploadProgressEvent | HttpResponse<any> =>\n        event.type === HttpEventType.Response || (enableProgress && event.type === HttpEventType.UploadProgress)\n      ),\n      map((event): UploadEvent => {\n        switch (event.type) {\n          case HttpEventType.UploadProgress: {\n            const { loaded = 0, total = 0 } = event;\n            return { type: 'uploadProgress', loaded, total, progress: total > 0 ? Math.round((loaded / total) * 100) : 0 };\n          }\n          case HttpEventType.Response:\n            return { type: 'response', body: event.body };\n          default:\n            throw new Error(\"Unexpected event type\"); // Should not reach here, but for typescript sake.\n        }\n      }),\n      catchError((error) => {\n        throw new Error(`Error invoking documentsupload: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Retrieves the indexing status of a predefined uploaded document(s)\n   * based on the provided status token returned from the upload process.\n   *\n   * @param token - The status token used to track the indexing process.\n   * @returns An Observable that emits the server response of type `IndexingResponse`.\n   * The `IndexingResponse` object contains:\n   * - `docs`: array - An array of document status objects, each containing:\n   *   - * `id`: string - The document ID.\n   *   - * `fileName`: string - The name of the file.\n   *   - * `operation`: \"Add\" | \"Update\" - The operation performed on the document.\n   *   - * `status`: \"Indexing\" | \"Indexed\" | \"Error\" - The status of the document.\n   *   - * `previousIndexationTime`: string | null - The previous indexation time.\n   *   - * `currentIndexationTime`: string | null - The current indexation time.\n   * - `executionTime`: string - The time taken to retrieve the indexing status.\n   * @throws {Error} If there is an error invoking the `documentsUploadStatus` action.\n   */\n  public getIndexingStatus(token: string): Observable<IndexingInfos> {\n    const data = {\n      action: \"documentsUploadStatus\",\n      appName: this.appService.appName,\n      statusToken: token\n    };\n\n    return from(post<IndexingInfos>(this.REQUEST_URL, data)).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsUploadStatus: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Retrieves a list of uploaded documents.\n   *\n   * @param {string[]} [columns] - Optional array of additional index columns to include for each document.\n   * @param {number} [skip] - Optional number of documents to skip for pagination. If set, count also needs to be set. No skip if not set.\n   * @param {number} [count] - Optional number of documents to take for pagination (unlimited if not set).\n   * @returns {Observable<UploadedDocuments>} An observable that emits the list of uploaded documents.\n   * The `UploadedDocuments` object contains:\n   * - `docs`: array - An array of objects representing a single document with the following properties:\n   *   - * `id`: string - A unique identifier for the document.\n   *   - * `fileName`: string - The name of the file.\n   *   - * `title`: string - The title of the document.\n   *   - * `fileExt`: string - The file extension of the document.\n   *   - * `indexationTime`: string - The time when the document was indexed.\n   *   - * `size`: number - The size of the document.\n   *   - * `sizeDisplay`: string - A human-readable representation of the document size.\n   *   - * `[key: string]`: any - Additional columns asked in the request, value of the column for the document.\n   * - `count`: number - The count of documents in the current batch or subset.\n   * - `totalCount`: number - The total count of documents available.\n   * - `executionTime`: string - The time taken to process the documents.\n   * @throws {Error} Throws an error if the request fails.\n   */\n  public getDocumentsList(columns?: string[], skip?: number, count?: number): Observable<UploadedDocuments> {\n    const data = {\n      action: \"documentsList\",\n      appName: this.appService.appName\n    };\n\n    if (columns) data[\"columns\"] = columns;\n    if (skip) data[\"skip\"] = skip;\n    if (count) data[\"count\"] = count;\n\n    return from(post<UploadedDocuments>(this.REQUEST_URL, data)).pipe(\n      tap((res: UploadedDocuments) => this.uploadedDocuments$.next(res.docs)),\n      catchError((error) => {\n        throw new Error(`Error invoking documentsList: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Deletes the uploaded documents with the specified IDs.\n   *\n   * @param docIds - An array of document IDs to delete.\n   * @returns An observable that emits the server response of type `deleteDocumentsResponse`.\n   * The `deleteDocumentsResponse` object contains:\n   * - `deletedCount`: number - The number of deleted documents.\n   * - `executionTime`: string - The time taken to delete the documents.\n   * @throws {Error} If there is an error invoking the `documentsDelete` action.\n   */\n  public deleteDocuments(docIds: string[]): Observable<deleteDocumentsResponse> {\n    const data = {\n      action: \"documentsDelete\",\n      docIds,\n      appName: this.appService.appName\n    };\n\n    return from(post<deleteDocumentsResponse>(this.REQUEST_URL, data)).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsDelete: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Deletes all uploaded documents.\n   *\n   * @returns An observable that emits the server response of type `deleteDocumentsResponse`.\n   * The `deleteDocumentsResponse` object contains:\n   * - `deletedCount`: number - The number of deleted documents.\n   * - `executionTime`: string - The time taken to delete the documents.\n   * @throws {Error} If there is an error invoking the `documentsDeleteAll` action.\n   */\n  public deleteAllDocuments(): Observable<deleteDocumentsResponse> {\n    const data = {\n      action: \"documentsDeleteAll\",\n      appName: this.appService.appName\n    };\n\n    return from(post<deleteDocumentsResponse>(this.REQUEST_URL, data)).pipe(\n      catchError((error) => {\n        throw new Error(`Error invoking documentsDeleteAll: ${error}`);\n      })\n    );\n  }\n\n  /**\n   * Formats the given file size in bytes into a human-readable string.\n   *\n   * @param bytes - The size of the file in bytes.\n   * @returns A string representing the file size in a human-readable format (e.g., \"10.24 KB\", \"1.00 MB\").\n   */\n  public formatFileSize(bytes: number): string {\n    if (bytes === 0) return \"0 Bytes\";\n    const k = 1024;\n    const sizes = [\"Bytes\", \"KB\", \"MB\", \"GB\", \"TB\"];\n    const i = Math.floor(Math.log(bytes) / Math.log(k));\n    return (\n      Number.parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + \" \" + sizes[i]\n    );\n  }\n\n  /**\n   * Converts the given file size in bytes into megabytes.\n   *\n   * @param bytes - The size of the file in bytes.\n   * @returns The size of the file in megabytes.\n   */\n  public convertBytesToMB(bytes: number): number {\n    return bytes / (1024 * 1024);\n  }\n\n}\n"]}
@@ -0,0 +1,14 @@
1
+ export function markdownItCodeBlockPlugin(md) {
2
+ const defaultFenceRenderer = md.renderer.rules.fence;
3
+ md.renderer.rules.fence = (tokens, idx, options, env, self) => {
4
+ const token = tokens[idx];
5
+ const info = token.info ? md.utils.unescapeAll(token.info).trim() : '';
6
+ const langName = info.split(/(\s+)/g)[0];
7
+ const defaultRendered = defaultFenceRenderer(tokens, idx, options, env, self);
8
+ // We add a wrapper and a copy button.
9
+ // The button will have an attribute `[copy-to-clipboard]` which can be handled by an Angular directive
10
+ // to implement the copy functionality. The code to be copied is passed as an input.
11
+ return `<code-block-reference langname="${langName}">${defaultRendered}</code-block-reference>`;
12
+ };
13
+ }
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29kZS1ibG9jay5wbHVnaW4uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hc3Npc3RhbnQvY2hhdC9tYXJrZG93bi1pdC1wbHVnaW5zL2NvZGUtYmxvY2sucGx1Z2luLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE1BQU0sVUFBVSx5QkFBeUIsQ0FBQyxFQUFjO0lBQ3RELE1BQU0sb0JBQW9CLEdBQUcsRUFBRSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsS0FBTSxDQUFDO0lBRXRELEVBQUUsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLEtBQUssR0FBRyxDQUFDLE1BQU0sRUFBRSxHQUFHLEVBQUUsT0FBTyxFQUFFLEdBQUcsRUFBRSxJQUFJLEVBQUUsRUFBRTtRQUM1RCxNQUFNLEtBQUssR0FBRyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDMUIsTUFBTSxJQUFJLEdBQUcsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7UUFDdkUsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN6QyxNQUFNLGVBQWUsR0FBRyxvQkFBb0IsQ0FBQyxNQUFNLEVBQUUsR0FBRyxFQUFFLE9BQU8sRUFBRSxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFFOUUsc0NBQXNDO1FBQ3RDLHVHQUF1RztRQUN2RyxvRkFBb0Y7UUFDcEYsT0FBTyxtQ0FBbUMsUUFBUSxLQUFLLGVBQWUseUJBQXlCLENBQUM7SUFDbEcsQ0FBQyxDQUFDO0FBQ0osQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIE1hcmtkb3duSXQgZnJvbSBcIm1hcmtkb3duLWl0XCI7XHJcblxyXG5leHBvcnQgZnVuY3Rpb24gbWFya2Rvd25JdENvZGVCbG9ja1BsdWdpbihtZDogTWFya2Rvd25JdCkge1xyXG4gIGNvbnN0IGRlZmF1bHRGZW5jZVJlbmRlcmVyID0gbWQucmVuZGVyZXIucnVsZXMuZmVuY2UhO1xyXG5cclxuICBtZC5yZW5kZXJlci5ydWxlcy5mZW5jZSA9ICh0b2tlbnMsIGlkeCwgb3B0aW9ucywgZW52LCBzZWxmKSA9PiB7XHJcbiAgICBjb25zdCB0b2tlbiA9IHRva2Vuc1tpZHhdO1xyXG4gICAgY29uc3QgaW5mbyA9IHRva2VuLmluZm8gPyBtZC51dGlscy51bmVzY2FwZUFsbCh0b2tlbi5pbmZvKS50cmltKCkgOiAnJztcclxuICAgIGNvbnN0IGxhbmdOYW1lID0gaW5mby5zcGxpdCgvKFxccyspL2cpWzBdO1xyXG4gICAgY29uc3QgZGVmYXVsdFJlbmRlcmVkID0gZGVmYXVsdEZlbmNlUmVuZGVyZXIodG9rZW5zLCBpZHgsIG9wdGlvbnMsIGVudiwgc2VsZik7XHJcblxyXG4gICAgLy8gV2UgYWRkIGEgd3JhcHBlciBhbmQgYSBjb3B5IGJ1dHRvbi5cclxuICAgIC8vIFRoZSBidXR0b24gd2lsbCBoYXZlIGFuIGF0dHJpYnV0ZSBgW2NvcHktdG8tY2xpcGJvYXJkXWAgd2hpY2ggY2FuIGJlIGhhbmRsZWQgYnkgYW4gQW5ndWxhciBkaXJlY3RpdmVcclxuICAgIC8vIHRvIGltcGxlbWVudCB0aGUgY29weSBmdW5jdGlvbmFsaXR5LiBUaGUgY29kZSB0byBiZSBjb3BpZWQgaXMgcGFzc2VkIGFzIGFuIGlucHV0LlxyXG4gICAgcmV0dXJuIGA8Y29kZS1ibG9jay1yZWZlcmVuY2UgbGFuZ25hbWU9XCIke2xhbmdOYW1lfVwiPiR7ZGVmYXVsdFJlbmRlcmVkfTwvY29kZS1ibG9jay1yZWZlcmVuY2U+YDtcclxuICB9O1xyXG59XHJcbiJdfQ==